ffplay.c 76.4 KB
Newer Older
bellard's avatar
bellard committed
1 2 3 4
/*
 * FFplay : Simple Media Player based on the ffmpeg libraries
 * Copyright (c) 2003 Fabrice Bellard
 *
5 6 7
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
bellard's avatar
bellard committed
8 9
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
10
 * version 2.1 of the License, or (at your option) any later version.
bellard's avatar
bellard committed
11
 *
12
 * FFmpeg is distributed in the hope that it will be useful,
bellard's avatar
bellard committed
13 14 15 16 17
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
18
 * License along with FFmpeg; if not, write to the Free Software
19
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
bellard's avatar
bellard committed
20
 */
21

22 23
#include <math.h>
#include <limits.h>
bellard's avatar
bellard committed
24
#include "avformat.h"
25
#include "swscale.h"
26
#include "avstring.h"
bellard's avatar
bellard committed
27

28
#include "version.h"
bellard's avatar
bellard committed
29 30 31 32 33
#include "cmdutils.h"

#include <SDL.h>
#include <SDL_thread.h>

34
#ifdef __MINGW32__
35 36 37
#undef main /* We don't want SDL to override our main() */
#endif

michael's avatar
michael committed
38 39
#undef exit

40 41
//#define DEBUG_SYNC

bellard's avatar
bellard committed
42 43
#define MAX_VIDEOQ_SIZE (5 * 256 * 1024)
#define MAX_AUDIOQ_SIZE (5 * 16 * 1024)
44
#define MAX_SUBTITLEQ_SIZE (5 * 16 * 1024)
bellard's avatar
bellard committed
45

46 47 48 49 50
/* SDL audio buffer size, in samples. Should be small to have precise
   A/V sync as SDL does not have hardware buffer fullness info. */
#define SDL_AUDIO_BUFFER_SIZE 1024

/* no AV sync correction is done if below the AV sync threshold */
51
#define AV_SYNC_THRESHOLD 0.01
52 53 54 55 56 57 58 59 60
/* no AV correction is done if too big error */
#define AV_NOSYNC_THRESHOLD 10.0

/* maximum audio speed change to get correct sync */
#define SAMPLE_CORRECTION_PERCENT_MAX 10

/* we use about AUDIO_DIFF_AVG_NB A-V differences to make the average */
#define AUDIO_DIFF_AVG_NB   20

bellard's avatar
bellard committed
61 62 63
/* NOTE: the size must be big enough to compensate the hardware audio buffersize size */
#define SAMPLE_ARRAY_SIZE (2*65536)

64 65
static int sws_flags = SWS_BICUBIC;

bellard's avatar
bellard committed
66 67 68 69 70 71 72 73 74 75
typedef struct PacketQueue {
    AVPacketList *first_pkt, *last_pkt;
    int nb_packets;
    int size;
    int abort_request;
    SDL_mutex *mutex;
    SDL_cond *cond;
} PacketQueue;

#define VIDEO_PICTURE_QUEUE_SIZE 1
76
#define SUBPICTURE_QUEUE_SIZE 4
bellard's avatar
bellard committed
77 78

typedef struct VideoPicture {
michael's avatar
michael committed
79
    double pts;                                  ///<presentation time stamp for this picture
bellard's avatar
bellard committed
80 81 82 83 84
    SDL_Overlay *bmp;
    int width, height; /* source height & width */
    int allocated;
} VideoPicture;

85 86 87 88 89
typedef struct SubPicture {
    double pts; /* presentation time stamp for this picture */
    AVSubtitle sub;
} SubPicture;

bellard's avatar
bellard committed
90 91 92
enum {
    AV_SYNC_AUDIO_MASTER, /* default choice */
    AV_SYNC_VIDEO_MASTER,
93
    AV_SYNC_EXTERNAL_CLOCK, /* synchronize to an external clock */
bellard's avatar
bellard committed
94 95 96 97 98
};

typedef struct VideoState {
    SDL_Thread *parse_tid;
    SDL_Thread *video_tid;
99
    AVInputFormat *iformat;
bellard's avatar
bellard committed
100 101 102
    int no_background;
    int abort_request;
    int paused;
103
    int last_paused;
bellard's avatar
bellard committed
104
    int seek_req;
michael's avatar
michael committed
105
    int seek_flags;
bellard's avatar
bellard committed
106
    int64_t seek_pos;
bellard's avatar
bellard committed
107 108 109 110
    AVFormatContext *ic;
    int dtg_active_format;

    int audio_stream;
111

bellard's avatar
bellard committed
112
    int av_sync_type;
113 114
    double external_clock; /* external clock base */
    int64_t external_clock_time;
115

116 117 118 119 120
    double audio_clock;
    double audio_diff_cum; /* used for AV difference average computation */
    double audio_diff_avg_coef;
    double audio_diff_threshold;
    int audio_diff_avg_count;
bellard's avatar
bellard committed
121 122 123 124 125
    AVStream *audio_st;
    PacketQueue audioq;
    int audio_hw_buf_size;
    /* samples output by the codec. we reserve more space for avsync
       compensation */
126
    DECLARE_ALIGNED(16,uint8_t,audio_buf[(AVCODEC_MAX_AUDIO_FRAME_SIZE * 3) / 2]);
127
    unsigned int audio_buf_size; /* in bytes */
bellard's avatar
bellard committed
128 129 130 131
    int audio_buf_index; /* in bytes */
    AVPacket audio_pkt;
    uint8_t *audio_pkt_data;
    int audio_pkt_size;
132

bellard's avatar
bellard committed
133 134 135
    int show_audio; /* if true, display audio samples */
    int16_t sample_array[SAMPLE_ARRAY_SIZE];
    int sample_array_index;
136
    int last_i_start;
137

138 139 140 141 142 143 144 145 146
    SDL_Thread *subtitle_tid;
    int subtitle_stream;
    int subtitle_stream_changed;
    AVStream *subtitle_st;
    PacketQueue subtitleq;
    SubPicture subpq[SUBPICTURE_QUEUE_SIZE];
    int subpq_size, subpq_rindex, subpq_windex;
    SDL_mutex *subpq_mutex;
    SDL_cond *subpq_cond;
147

148 149 150
    double frame_timer;
    double frame_last_pts;
    double frame_last_delay;
151
    double video_clock;                          ///<pts of last decoded frame / predicted pts of next decoded frame
bellard's avatar
bellard committed
152 153 154
    int video_stream;
    AVStream *video_st;
    PacketQueue videoq;
michael's avatar
michael committed
155 156
    double video_current_pts;                    ///<current displayed pts (different from video_clock if frame fifos are used)
    int64_t video_current_pts_time;              ///<time (av_gettime) at which we updated video_current_pts - used to have running video pts
bellard's avatar
bellard committed
157 158 159 160
    VideoPicture pictq[VIDEO_PICTURE_QUEUE_SIZE];
    int pictq_size, pictq_rindex, pictq_windex;
    SDL_mutex *pictq_mutex;
    SDL_cond *pictq_cond;
161

bellard's avatar
bellard committed
162 163 164 165 166 167
    //    QETimer *video_timer;
    char filename[1024];
    int width, height, xleft, ytop;
} VideoState;

void show_help(void);
168
static int audio_write_get_buf_size(VideoState *is);
bellard's avatar
bellard committed
169 170 171 172 173 174

/* options specified by the user */
static AVInputFormat *file_iformat;
static const char *input_filename;
static int fs_screen_width;
static int fs_screen_height;
michael's avatar
michael committed
175 176
static int screen_width = 0;
static int screen_height = 0;
177 178 179
static int frame_width = 0;
static int frame_height = 0;
static enum PixelFormat frame_pix_fmt = PIX_FMT_NONE;
bellard's avatar
bellard committed
180 181
static int audio_disable;
static int video_disable;
182
static int wanted_audio_stream= 0;
183
static int wanted_video_stream= 0;
184
static int seek_by_bytes;
bellard's avatar
bellard committed
185 186
static int display_disable;
static int show_status;
187
static int av_sync_type = AV_SYNC_AUDIO_MASTER;
bellard's avatar
bellard committed
188
static int64_t start_time = AV_NOPTS_VALUE;
189
static int debug = 0;
190
static int debug_mv = 0;
191
static int step = 0;
michael's avatar
michael committed
192
static int thread_count = 1;
michael's avatar
 
michael committed
193
static int workaround_bugs = 1;
194
static int fast = 0;
195
static int genpts = 0;
michael's avatar
michael committed
196 197
static int lowres = 0;
static int idct = FF_IDCT_AUTO;
michael's avatar
michael committed
198 199 200
static enum AVDiscard skip_frame= AVDISCARD_DEFAULT;
static enum AVDiscard skip_idct= AVDISCARD_DEFAULT;
static enum AVDiscard skip_loop_filter= AVDISCARD_DEFAULT;
diego's avatar
diego committed
201
static int error_resilience = FF_ER_CAREFUL;
202
static int error_concealment = 3;
203
static int decoder_reorder_pts= 0;
bellard's avatar
bellard committed
204 205 206 207

/* current context */
static int is_full_screen;
static VideoState *cur_stream;
208
static int64_t audio_callback_time;
bellard's avatar
bellard committed
209

michael's avatar
michael committed
210 211
AVPacket flush_pkt;

bellard's avatar
bellard committed
212 213
#define FF_ALLOC_EVENT   (SDL_USEREVENT)
#define FF_REFRESH_EVENT (SDL_USEREVENT + 1)
214
#define FF_QUIT_EVENT    (SDL_USEREVENT + 2)
bellard's avatar
bellard committed
215 216 217 218 219 220 221 222 223 224 225

SDL_Surface *screen;

/* packet queue handling */
static void packet_queue_init(PacketQueue *q)
{
    memset(q, 0, sizeof(PacketQueue));
    q->mutex = SDL_CreateMutex();
    q->cond = SDL_CreateCond();
}

bellard's avatar
bellard committed
226
static void packet_queue_flush(PacketQueue *q)
bellard's avatar
bellard committed
227 228 229
{
    AVPacketList *pkt, *pkt1;

230
    SDL_LockMutex(q->mutex);
bellard's avatar
bellard committed
231 232 233
    for(pkt = q->first_pkt; pkt != NULL; pkt = pkt1) {
        pkt1 = pkt->next;
        av_free_packet(&pkt->pkt);
234
        av_freep(&pkt);
bellard's avatar
bellard committed
235
    }
bellard's avatar
bellard committed
236 237 238 239
    q->last_pkt = NULL;
    q->first_pkt = NULL;
    q->nb_packets = 0;
    q->size = 0;
240
    SDL_UnlockMutex(q->mutex);
bellard's avatar
bellard committed
241 242 243 244 245
}

static void packet_queue_end(PacketQueue *q)
{
    packet_queue_flush(q);
bellard's avatar
bellard committed
246 247 248 249 250 251 252 253
    SDL_DestroyMutex(q->mutex);
    SDL_DestroyCond(q->cond);
}

static int packet_queue_put(PacketQueue *q, AVPacket *pkt)
{
    AVPacketList *pkt1;

bellard's avatar
bellard committed
254
    /* duplicate the packet */
michael's avatar
michael committed
255
    if (pkt!=&flush_pkt && av_dup_packet(pkt) < 0)
bellard's avatar
bellard committed
256
        return -1;
257

bellard's avatar
bellard committed
258 259 260 261 262 263
    pkt1 = av_malloc(sizeof(AVPacketList));
    if (!pkt1)
        return -1;
    pkt1->pkt = *pkt;
    pkt1->next = NULL;

bellard's avatar
bellard committed
264

bellard's avatar
bellard committed
265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286
    SDL_LockMutex(q->mutex);

    if (!q->last_pkt)

        q->first_pkt = pkt1;
    else
        q->last_pkt->next = pkt1;
    q->last_pkt = pkt1;
    q->nb_packets++;
    q->size += pkt1->pkt.size;
    /* XXX: should duplicate packet data in DV case */
    SDL_CondSignal(q->cond);

    SDL_UnlockMutex(q->mutex);
    return 0;
}

static void packet_queue_abort(PacketQueue *q)
{
    SDL_LockMutex(q->mutex);

    q->abort_request = 1;
287

bellard's avatar
bellard committed
288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305
    SDL_CondSignal(q->cond);

    SDL_UnlockMutex(q->mutex);
}

/* return < 0 if aborted, 0 if no packet and > 0 if packet.  */
static int packet_queue_get(PacketQueue *q, AVPacket *pkt, int block)
{
    AVPacketList *pkt1;
    int ret;

    SDL_LockMutex(q->mutex);

    for(;;) {
        if (q->abort_request) {
            ret = -1;
            break;
        }
306

bellard's avatar
bellard committed
307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328
        pkt1 = q->first_pkt;
        if (pkt1) {
            q->first_pkt = pkt1->next;
            if (!q->first_pkt)
                q->last_pkt = NULL;
            q->nb_packets--;
            q->size -= pkt1->pkt.size;
            *pkt = pkt1->pkt;
            av_free(pkt1);
            ret = 1;
            break;
        } else if (!block) {
            ret = 0;
            break;
        } else {
            SDL_CondWait(q->cond, q->mutex);
        }
    }
    SDL_UnlockMutex(q->mutex);
    return ret;
}

329
static inline void fill_rectangle(SDL_Surface *screen,
bellard's avatar
bellard committed
330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358
                                  int x, int y, int w, int h, int color)
{
    SDL_Rect rect;
    rect.x = x;
    rect.y = y;
    rect.w = w;
    rect.h = h;
    SDL_FillRect(screen, &rect, color);
}

#if 0
/* draw only the border of a rectangle */
void fill_border(VideoState *s, int x, int y, int w, int h, int color)
{
    int w1, w2, h1, h2;

    /* fill the background */
    w1 = x;
    if (w1 < 0)
        w1 = 0;
    w2 = s->width - (x + w);
    if (w2 < 0)
        w2 = 0;
    h1 = y;
    if (h1 < 0)
        h1 = 0;
    h2 = s->height - (y + h);
    if (h2 < 0)
        h2 = 0;
359 360 361
    fill_rectangle(screen,
                   s->xleft, s->ytop,
                   w1, s->height,
bellard's avatar
bellard committed
362
                   color);
363 364 365
    fill_rectangle(screen,
                   s->xleft + s->width - w2, s->ytop,
                   w2, s->height,
bellard's avatar
bellard committed
366
                   color);
367 368 369
    fill_rectangle(screen,
                   s->xleft + w1, s->ytop,
                   s->width - w1 - w2, h1,
bellard's avatar
bellard committed
370
                   color);
371
    fill_rectangle(screen,
bellard's avatar
bellard committed
372 373 374 375 376 377
                   s->xleft + w1, s->ytop + s->height - h2,
                   s->width - w1 - w2, h2,
                   color);
}
#endif

378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409


#define SCALEBITS 10
#define ONE_HALF  (1 << (SCALEBITS - 1))
#define FIX(x)    ((int) ((x) * (1<<SCALEBITS) + 0.5))

#define RGB_TO_Y_CCIR(r, g, b) \
((FIX(0.29900*219.0/255.0) * (r) + FIX(0.58700*219.0/255.0) * (g) + \
  FIX(0.11400*219.0/255.0) * (b) + (ONE_HALF + (16 << SCALEBITS))) >> SCALEBITS)

#define RGB_TO_U_CCIR(r1, g1, b1, shift)\
(((- FIX(0.16874*224.0/255.0) * r1 - FIX(0.33126*224.0/255.0) * g1 +         \
     FIX(0.50000*224.0/255.0) * b1 + (ONE_HALF << shift) - 1) >> (SCALEBITS + shift)) + 128)

#define RGB_TO_V_CCIR(r1, g1, b1, shift)\
(((FIX(0.50000*224.0/255.0) * r1 - FIX(0.41869*224.0/255.0) * g1 -           \
   FIX(0.08131*224.0/255.0) * b1 + (ONE_HALF << shift) - 1) >> (SCALEBITS + shift)) + 128)

#define ALPHA_BLEND(a, oldp, newp, s)\
((((oldp << s) * (255 - (a))) + (newp * (a))) / (255 << s))

#define RGBA_IN(r, g, b, a, s)\
{\
    unsigned int v = ((const uint32_t *)(s))[0];\
    a = (v >> 24) & 0xff;\
    r = (v >> 16) & 0xff;\
    g = (v >> 8) & 0xff;\
    b = v & 0xff;\
}

#define YUVA_IN(y, u, v, a, s, pal)\
{\
410
    unsigned int val = ((const uint32_t *)(pal))[*(const uint8_t*)(s)];\
411 412 413 414 415 416 417 418 419 420 421 422 423 424
    a = (val >> 24) & 0xff;\
    y = (val >> 16) & 0xff;\
    u = (val >> 8) & 0xff;\
    v = val & 0xff;\
}

#define YUVA_OUT(d, y, u, v, a)\
{\
    ((uint32_t *)(d))[0] = (a << 24) | (y << 16) | (u << 8) | v;\
}


#define BPP 1

425
static void blend_subrect(AVPicture *dst, const AVSubtitleRect *rect, int imgw, int imgh)
426 427 428 429 430 431
{
    int wrap, wrap3, width2, skip2;
    int y, u, v, a, u1, v1, a1, w, h;
    uint8_t *lum, *cb, *cr;
    const uint8_t *p;
    const uint32_t *pal;
432 433
    int dstx, dsty, dstw, dsth;

434
    dstx = FFMIN(FFMAX(rect->x, 0), imgw);
435
    dstw = FFMIN(FFMAX(rect->w, 0), imgw - dstx);
436
    dsty = FFMIN(FFMAX(rect->y, 0), imgh);
437 438 439 440 441 442 443
    dsth = FFMIN(FFMAX(rect->h, 0), imgh - dsty);
    lum = dst->data[0] + dsty * dst->linesize[0];
    cb = dst->data[1] + (dsty >> 1) * dst->linesize[1];
    cr = dst->data[2] + (dsty >> 1) * dst->linesize[2];

    width2 = (dstw + 1) >> 1;
    skip2 = dstx >> 1;
444 445 446 447
    wrap = dst->linesize[0];
    wrap3 = rect->linesize;
    p = rect->bitmap;
    pal = rect->rgba_palette;  /* Now in YCrCb! */
448

449 450
    if (dsty & 1) {
        lum += dstx;
451 452
        cb += skip2;
        cr += skip2;
453

454
        if (dstx & 1) {
455 456 457 458 459 460 461 462 463
            YUVA_IN(y, u, v, a, p, pal);
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
            cb++;
            cr++;
            lum++;
            p += BPP;
        }
464
        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488
            YUVA_IN(y, u, v, a, p, pal);
            u1 = u;
            v1 = v;
            a1 = a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);

            YUVA_IN(y, u, v, a, p + BPP, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
            cb++;
            cr++;
            p += 2 * BPP;
            lum += 2;
        }
        if (w) {
            YUVA_IN(y, u, v, a, p, pal);
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
        }
489 490
        p += wrap3 + (wrap3 - dstw * BPP);
        lum += wrap + (wrap - dstw - dstx);
491 492 493
        cb += dst->linesize[1] - width2 - skip2;
        cr += dst->linesize[2] - width2 - skip2;
    }
494 495
    for(h = dsth - (dsty & 1); h >= 2; h -= 2) {
        lum += dstx;
496 497
        cb += skip2;
        cr += skip2;
498

499
        if (dstx & 1) {
500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518
            YUVA_IN(y, u, v, a, p, pal);
            u1 = u;
            v1 = v;
            a1 = a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            p += wrap3;
            lum += wrap;
            YUVA_IN(y, u, v, a, p, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
            cb++;
            cr++;
            p += -wrap3 + BPP;
            lum += -wrap + 1;
        }
519
        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573
            YUVA_IN(y, u, v, a, p, pal);
            u1 = u;
            v1 = v;
            a1 = a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);

            YUVA_IN(y, u, v, a, p, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
            p += wrap3;
            lum += wrap;

            YUVA_IN(y, u, v, a, p, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);

            YUVA_IN(y, u, v, a, p, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);

            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 2);
            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 2);

            cb++;
            cr++;
            p += -wrap3 + 2 * BPP;
            lum += -wrap + 2;
        }
        if (w) {
            YUVA_IN(y, u, v, a, p, pal);
            u1 = u;
            v1 = v;
            a1 = a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            p += wrap3;
            lum += wrap;
            YUVA_IN(y, u, v, a, p, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
            cb++;
            cr++;
            p += -wrap3 + BPP;
            lum += -wrap + 1;
        }
574 575
        p += wrap3 + (wrap3 - dstw * BPP);
        lum += wrap + (wrap - dstw - dstx);
576 577 578 579 580
        cb += dst->linesize[1] - width2 - skip2;
        cr += dst->linesize[2] - width2 - skip2;
    }
    /* handle odd height */
    if (h) {
581
        lum += dstx;
582 583
        cb += skip2;
        cr += skip2;
584

585
        if (dstx & 1) {
586 587 588 589 590 591 592 593 594
            YUVA_IN(y, u, v, a, p, pal);
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
            cb++;
            cr++;
            lum++;
            p += BPP;
        }
595
        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625
            YUVA_IN(y, u, v, a, p, pal);
            u1 = u;
            v1 = v;
            a1 = a;
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);

            YUVA_IN(y, u, v, a, p + BPP, pal);
            u1 += u;
            v1 += v;
            a1 += a;
            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u, 1);
            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v, 1);
            cb++;
            cr++;
            p += 2 * BPP;
            lum += 2;
        }
        if (w) {
            YUVA_IN(y, u, v, a, p, pal);
            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
        }
    }
}

static void free_subpicture(SubPicture *sp)
{
    int i;
626

627 628 629 630 631
    for (i = 0; i < sp->sub.num_rects; i++)
    {
        av_free(sp->sub.rects[i].bitmap);
        av_free(sp->sub.rects[i].rgba_palette);
    }
632

633
    av_free(sp->sub.rects);
634

635 636 637
    memset(&sp->sub, 0, sizeof(AVSubtitle));
}

bellard's avatar
bellard committed
638 639 640
static void video_image_display(VideoState *is)
{
    VideoPicture *vp;
641 642
    SubPicture *sp;
    AVPicture pict;
bellard's avatar
bellard committed
643 644 645
    float aspect_ratio;
    int width, height, x, y;
    SDL_Rect rect;
646
    int i;
bellard's avatar
bellard committed
647 648 649 650

    vp = &is->pictq[is->pictq_rindex];
    if (vp->bmp) {
        /* XXX: use variable in the frame */
651
        if (is->video_st->codec->sample_aspect_ratio.num == 0)
bellard's avatar
bellard committed
652 653
            aspect_ratio = 0;
        else
654
            aspect_ratio = av_q2d(is->video_st->codec->sample_aspect_ratio)
655
                * is->video_st->codec->width / is->video_st->codec->height;;
bellard's avatar
bellard committed
656
        if (aspect_ratio <= 0.0)
657
            aspect_ratio = (float)is->video_st->codec->width /
658
                (float)is->video_st->codec->height;
bellard's avatar
bellard committed
659 660 661
        /* if an active format is indicated, then it overrides the
           mpeg format */
#if 0
662 663
        if (is->video_st->codec->dtg_active_format != is->dtg_active_format) {
            is->dtg_active_format = is->video_st->codec->dtg_active_format;
bellard's avatar
bellard committed
664 665 666 667
            printf("dtg_active_format=%d\n", is->dtg_active_format);
        }
#endif
#if 0
668
        switch(is->video_st->codec->dtg_active_format) {
bellard's avatar
bellard committed
669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693
        case FF_DTG_AFD_SAME:
        default:
            /* nothing to do */
            break;
        case FF_DTG_AFD_4_3:
            aspect_ratio = 4.0 / 3.0;
            break;
        case FF_DTG_AFD_16_9:
            aspect_ratio = 16.0 / 9.0;
            break;
        case FF_DTG_AFD_14_9:
            aspect_ratio = 14.0 / 9.0;
            break;
        case FF_DTG_AFD_4_3_SP_14_9:
            aspect_ratio = 14.0 / 9.0;
            break;
        case FF_DTG_AFD_16_9_SP_14_9:
            aspect_ratio = 14.0 / 9.0;
            break;
        case FF_DTG_AFD_SP_4_3:
            aspect_ratio = 4.0 / 3.0;
            break;
        }
#endif

694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712
        if (is->subtitle_st)
        {
            if (is->subpq_size > 0)
            {
                sp = &is->subpq[is->subpq_rindex];

                if (vp->pts >= sp->pts + ((float) sp->sub.start_display_time / 1000))
                {
                    SDL_LockYUVOverlay (vp->bmp);

                    pict.data[0] = vp->bmp->pixels[0];
                    pict.data[1] = vp->bmp->pixels[2];
                    pict.data[2] = vp->bmp->pixels[1];

                    pict.linesize[0] = vp->bmp->pitches[0];
                    pict.linesize[1] = vp->bmp->pitches[2];
                    pict.linesize[2] = vp->bmp->pitches[1];

                    for (i = 0; i < sp->sub.num_rects; i++)
713
                        blend_subrect(&pict, &sp->sub.rects[i],
714
                                      vp->bmp->w, vp->bmp->h);
715 716 717 718 719 720 721

                    SDL_UnlockYUVOverlay (vp->bmp);
                }
            }
        }


bellard's avatar
bellard committed
722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737
        /* XXX: we suppose the screen has a 1.0 pixel ratio */
        height = is->height;
        width = ((int)rint(height * aspect_ratio)) & -3;
        if (width > is->width) {
            width = is->width;
            height = ((int)rint(width / aspect_ratio)) & -3;
        }
        x = (is->width - width) / 2;
        y = (is->height - height) / 2;
        if (!is->no_background) {
            /* fill the background */
            //            fill_border(is, x, y, width, height, QERGB(0x00, 0x00, 0x00));
        } else {
            is->no_background = 0;
        }
        rect.x = is->xleft + x;
bcoudurier's avatar
bcoudurier committed
738
        rect.y = is->ytop  + y;
bellard's avatar
bellard committed
739 740 741 742 743
        rect.w = width;
        rect.h = height;
        SDL_DisplayYUVOverlay(vp->bmp, &rect);
    } else {
#if 0
744 745
        fill_rectangle(screen,
                       is->xleft, is->ytop, is->width, is->height,
bellard's avatar
bellard committed
746 747 748 749 750 751 752 753
                       QERGB(0x00, 0x00, 0x00));
#endif
    }
}

static inline int compute_mod(int a, int b)
{
    a = a % b;
754
    if (a >= 0)
bellard's avatar
bellard committed
755 756 757 758 759 760 761 762 763 764
        return a;
    else
        return a + b;
}

static void video_audio_display(VideoState *s)
{
    int i, i_start, x, y1, y, ys, delay, n, nb_display_channels;
    int ch, channels, h, h2, bgcolor, fgcolor;
    int16_t time_diff;
765

bellard's avatar
bellard committed
766
    /* compute display index : center on currently output samples */
767
    channels = s->audio_st->codec->channels;
bellard's avatar
bellard committed
768
    nb_display_channels = channels;
769 770 771 772
    if (!s->paused) {
        n = 2 * channels;
        delay = audio_write_get_buf_size(s);
        delay /= n;
773

774 775 776 777
        /* to be more precise, we take into account the time spent since
           the last buffer computation */
        if (audio_callback_time) {
            time_diff = av_gettime() - audio_callback_time;
778
            delay += (time_diff * s->audio_st->codec->sample_rate) / 1000000;
779
        }
780

781 782 783
        delay -= s->width / 2;
        if (delay < s->width)
            delay = s->width;
784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800

        i_start= x = compute_mod(s->sample_array_index - delay * channels, SAMPLE_ARRAY_SIZE);

        h= INT_MIN;
        for(i=0; i<1000; i+=channels){
            int idx= (SAMPLE_ARRAY_SIZE + x - i) % SAMPLE_ARRAY_SIZE;
            int a= s->sample_array[idx];
            int b= s->sample_array[(idx + 4*channels)%SAMPLE_ARRAY_SIZE];
            int c= s->sample_array[(idx + 5*channels)%SAMPLE_ARRAY_SIZE];
            int d= s->sample_array[(idx + 9*channels)%SAMPLE_ARRAY_SIZE];
            int score= a-d;
            if(h<score && (b^c)<0){
                h= score;
                i_start= idx;
            }
        }

801 802 803
        s->last_i_start = i_start;
    } else {
        i_start = s->last_i_start;
bellard's avatar
bellard committed
804 805 806
    }

    bgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0x00);
807 808
    fill_rectangle(screen,
                   s->xleft, s->ytop, s->width, s->height,
bellard's avatar
bellard committed
809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827
                   bgcolor);

    fgcolor = SDL_MapRGB(screen->format, 0xff, 0xff, 0xff);

    /* total height for one channel */
    h = s->height / nb_display_channels;
    /* graph height / 2 */
    h2 = (h * 9) / 20;
    for(ch = 0;ch < nb_display_channels; ch++) {
        i = i_start + ch;
        y1 = s->ytop + ch * h + (h / 2); /* position of center line */
        for(x = 0; x < s->width; x++) {
            y = (s->sample_array[i] * h2) >> 15;
            if (y < 0) {
                y = -y;
                ys = y1 - y;
            } else {
                ys = y1;
            }
828 829
            fill_rectangle(screen,
                           s->xleft + x, ys, 1, y,
bellard's avatar
bellard committed
830 831 832 833 834 835 836 837 838 839 840
                           fgcolor);
            i += channels;
            if (i >= SAMPLE_ARRAY_SIZE)
                i -= SAMPLE_ARRAY_SIZE;
        }
    }

    fgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0xff);

    for(ch = 1;ch < nb_display_channels; ch++) {
        y = s->ytop + ch * h;
841 842
        fill_rectangle(screen,
                       s->xleft, y, s->width, 1,
bellard's avatar
bellard committed
843 844 845 846 847
                       fgcolor);
    }
    SDL_UpdateRect(screen, s->xleft, s->ytop, s->width, s->height);
}

848 849 850 851
static int video_open(VideoState *is){
    int flags = SDL_HWSURFACE|SDL_ASYNCBLIT|SDL_HWACCEL;
    int w,h;

852 853 854
    if(is_full_screen) flags |= SDL_FULLSCREEN;
    else               flags |= SDL_RESIZABLE;

855 856 857
    if (is_full_screen && fs_screen_width) {
        w = fs_screen_width;
        h = fs_screen_height;
858 859 860 861 862 863
    } else if(!is_full_screen && screen_width){
        w = screen_width;
        h = screen_height;
    }else if (is->video_st && is->video_st->codec->width){
        w = is->video_st->codec->width;
        h = is->video_st->codec->height;
864
    } else {
865 866
        w = 640;
        h = 480;
867
    }
868
#ifndef __APPLE__
869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884
    screen = SDL_SetVideoMode(w, h, 0, flags);
#else
    /* setting bits_per_pixel = 0 or 32 causes blank video on OS X */
    screen = SDL_SetVideoMode(w, h, 24, flags);
#endif
    if (!screen) {
        fprintf(stderr, "SDL: could not set video mode - exiting\n");
        return -1;
    }
    SDL_WM_SetCaption("FFplay", "FFplay");

    is->width = screen->w;
    is->height = screen->h;

    return 0;
}
885

bellard's avatar
bellard committed
886 887 888
/* display the current picture, if any */
static void video_display(VideoState *is)
{
889 890
    if(!screen)
        video_open(cur_stream);
891
    if (is->audio_st && is->show_audio)
bellard's avatar
bellard committed
892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911
        video_audio_display(is);
    else if (is->video_st)
        video_image_display(is);
}

static Uint32 sdl_refresh_timer_cb(Uint32 interval, void *opaque)
{
    SDL_Event event;
    event.type = FF_REFRESH_EVENT;
    event.user.data1 = opaque;
    SDL_PushEvent(&event);
    return 0; /* 0 means stop timer */
}

/* schedule a video refresh in 'delay' ms */
static void schedule_refresh(VideoState *is, int delay)
{
    SDL_AddTimer(delay, sdl_refresh_timer_cb, is);
}

912 913 914 915 916 917 918 919 920
/* get the current audio clock value */
static double get_audio_clock(VideoState *is)
{
    double pts;
    int hw_buf_size, bytes_per_sec;
    pts = is->audio_clock;
    hw_buf_size = audio_write_get_buf_size(is);
    bytes_per_sec = 0;
    if (is->audio_st) {
921
        bytes_per_sec = is->audio_st->codec->sample_rate *
922
            2 * is->audio_st->codec->channels;
923 924 925 926 927 928 929 930 931 932
    }
    if (bytes_per_sec)
        pts -= (double)hw_buf_size / bytes_per_sec;
    return pts;
}

/* get the current video clock value */
static double get_video_clock(VideoState *is)
{
    double delta;
michael's avatar
oops  
michael committed
933
    if (is->paused) {
bellard's avatar
bellard committed
934 935 936 937
        delta = 0;
    } else {
        delta = (av_gettime() - is->video_current_pts_time) / 1000000.0;
    }
938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953
    return is->video_current_pts + delta;
}

/* get the current external clock value */
static double get_external_clock(VideoState *is)
{
    int64_t ti;
    ti = av_gettime();
    return is->external_clock + ((ti - is->external_clock_time) * 1e-6);
}

/* get the current master clock value */
static double get_master_clock(VideoState *is)
{
    double val;

bellard's avatar
bellard committed
954 955 956 957 958 959 960 961 962 963 964
    if (is->av_sync_type == AV_SYNC_VIDEO_MASTER) {
        if (is->video_st)
            val = get_video_clock(is);
        else
            val = get_audio_clock(is);
    } else if (is->av_sync_type == AV_SYNC_AUDIO_MASTER) {
        if (is->audio_st)
            val = get_audio_clock(is);
        else
            val = get_video_clock(is);
    } else {
965
        val = get_external_clock(is);
bellard's avatar
bellard committed
966
    }
967 968 969
    return val;
}

bellard's avatar
bellard committed
970
/* seek in the stream */
michael's avatar
michael committed
971
static void stream_seek(VideoState *is, int64_t pos, int rel)
bellard's avatar
bellard committed
972
{
973 974 975
    if (!is->seek_req) {
        is->seek_pos = pos;
        is->seek_flags = rel < 0 ? AVSEEK_FLAG_BACKWARD : 0;
976 977
        if (seek_by_bytes)
            is->seek_flags |= AVSEEK_FLAG_BYTE;
978 979
        is->seek_req = 1;
    }
bellard's avatar
bellard committed
980 981 982 983 984 985
}

/* pause or resume the video */
static void stream_pause(VideoState *is)
{
    is->paused = !is->paused;
986
    if (!is->paused) {
bellard's avatar
bellard committed
987
        is->video_current_pts = get_video_clock(is);
988
        is->frame_timer += (av_gettime() - is->video_current_pts_time) / 1000000.0;
bellard's avatar
bellard committed
989 990 991
    }
}

bellard's avatar
bellard committed
992 993 994 995 996
/* called to display each frame */
static void video_refresh_timer(void *opaque)
{
    VideoState *is = opaque;
    VideoPicture *vp;
997 998
    double actual_delay, delay, sync_threshold, ref_clock, diff;

999
    SubPicture *sp, *sp2;
bellard's avatar
bellard committed
1000 1001 1002 1003

    if (is->video_st) {
        if (is->pictq_size == 0) {
            /* if no picture, need to wait */
1004
            schedule_refresh(is, 1);
bellard's avatar
bellard committed
1005
        } else {
1006
            /* dequeue the picture */
bellard's avatar
bellard committed
1007
            vp = &is->pictq[is->pictq_rindex];
1008 1009 1010 1011 1012 1013 1014

            /* update current video pts */
            is->video_current_pts = vp->pts;
            is->video_current_pts_time = av_gettime();

            /* compute nominal delay */
            delay = vp->pts - is->frame_last_pts;
michael's avatar
michael committed
1015
            if (delay <= 0 || delay >= 2.0) {
1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028
                /* if incorrect delay, use previous one */
                delay = is->frame_last_delay;
            }
            is->frame_last_delay = delay;
            is->frame_last_pts = vp->pts;

            /* update delay to follow master synchronisation source */
            if (((is->av_sync_type == AV_SYNC_AUDIO_MASTER && is->audio_st) ||
                 is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
                /* if video is slave, we try to correct big delays by
                   duplicating or deleting a frame */
                ref_clock = get_master_clock(is);
                diff = vp->pts - ref_clock;
1029

1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051
                /* skip or repeat frame. We take into account the
                   delay to compute the threshold. I still don't know
                   if it is the best guess */
                sync_threshold = AV_SYNC_THRESHOLD;
                if (delay > sync_threshold)
                    sync_threshold = delay;
                if (fabs(diff) < AV_NOSYNC_THRESHOLD) {
                    if (diff <= -sync_threshold)
                        delay = 0;
                    else if (diff >= sync_threshold)
                        delay = 2 * delay;
                }
            }

            is->frame_timer += delay;
            /* compute the REAL delay (we need to do that to avoid
               long term errors */
            actual_delay = is->frame_timer - (av_gettime() / 1000000.0);
            if (actual_delay < 0.010) {
                /* XXX: should skip picture */
                actual_delay = 0.010;
            }
bellard's avatar
bellard committed
1052
            /* launch timer for next picture */
1053 1054 1055
            schedule_refresh(is, (int)(actual_delay * 1000 + 0.5));

#if defined(DEBUG_SYNC)
1056
            printf("video: delay=%0.3f actual_delay=%0.3f pts=%0.3f A-V=%f\n",
1057 1058
                   delay, actual_delay, vp->pts, -diff);
#endif
bellard's avatar
bellard committed
1059

1060 1061 1062
            if(is->subtitle_st) {
                if (is->subtitle_stream_changed) {
                    SDL_LockMutex(is->subpq_mutex);
1063

1064 1065
                    while (is->subpq_size) {
                        free_subpicture(&is->subpq[is->subpq_rindex]);
1066

1067 1068 1069
                        /* update queue size and signal for next picture */
                        if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
                            is->subpq_rindex = 0;
1070

1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103
                        is->subpq_size--;
                    }
                    is->subtitle_stream_changed = 0;

                    SDL_CondSignal(is->subpq_cond);
                    SDL_UnlockMutex(is->subpq_mutex);
                } else {
                    if (is->subpq_size > 0) {
                        sp = &is->subpq[is->subpq_rindex];

                        if (is->subpq_size > 1)
                            sp2 = &is->subpq[(is->subpq_rindex + 1) % SUBPICTURE_QUEUE_SIZE];
                        else
                            sp2 = NULL;

                        if ((is->video_current_pts > (sp->pts + ((float) sp->sub.end_display_time / 1000)))
                                || (sp2 && is->video_current_pts > (sp2->pts + ((float) sp2->sub.start_display_time / 1000))))
                        {
                            free_subpicture(sp);

                            /* update queue size and signal for next picture */
                            if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
                                is->subpq_rindex = 0;

                            SDL_LockMutex(is->subpq_mutex);
                            is->subpq_size--;
                            SDL_CondSignal(is->subpq_cond);
                            SDL_UnlockMutex(is->subpq_mutex);
                        }
                    }
                }
            }

bellard's avatar
bellard committed
1104 1105
            /* display picture */
            video_display(is);
1106

bellard's avatar
bellard committed
1107 1108 1109
            /* update queue size and signal for next picture */
            if (++is->pictq_rindex == VIDEO_PICTURE_QUEUE_SIZE)
                is->pictq_rindex = 0;
1110

bellard's avatar
bellard committed
1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122
            SDL_LockMutex(is->pictq_mutex);
            is->pictq_size--;
            SDL_CondSignal(is->pictq_cond);
            SDL_UnlockMutex(is->pictq_mutex);
        }
    } else if (is->audio_st) {
        /* draw the next audio frame */

        schedule_refresh(is, 40);

        /* if only audio stream, then display the audio bars (better
           than nothing, just to test the implementation */
1123

bellard's avatar
bellard committed
1124 1125 1126 1127 1128 1129 1130 1131
        /* display picture */
        video_display(is);
    } else {
        schedule_refresh(is, 100);
    }
    if (show_status) {
        static int64_t last_time;
        int64_t cur_time;
1132
        int aqsize, vqsize, sqsize;
1133
        double av_diff;
1134

bellard's avatar
bellard committed
1135 1136 1137 1138
        cur_time = av_gettime();
        if (!last_time || (cur_time - last_time) >= 500 * 1000) {
            aqsize = 0;
            vqsize = 0;
1139
            sqsize = 0;
bellard's avatar
bellard committed
1140 1141 1142 1143
            if (is->audio_st)
                aqsize = is->audioq.size;
            if (is->video_st)
                vqsize = is->videoq.size;
1144 1145
            if (is->subtitle_st)
                sqsize = is->subtitleq.size;
1146 1147 1148
            av_diff = 0;
            if (is->audio_st && is->video_st)
                av_diff = get_audio_clock(is) - get_video_clock(is);
1149
            printf("%7.2f A-V:%7.3f aq=%5dKB vq=%5dKB sq=%5dB    \r",
1150
                   get_master_clock(is), av_diff, aqsize / 1024, vqsize / 1024, sqsize);
bellard's avatar
bellard committed
1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168
            fflush(stdout);
            last_time = cur_time;
        }
    }
}

/* allocate a picture (needs to do that in main thread to avoid
   potential locking problems */
static void alloc_picture(void *opaque)
{
    VideoState *is = opaque;
    VideoPicture *vp;

    vp = &is->pictq[is->pictq_windex];

    if (vp->bmp)
        SDL_FreeYUVOverlay(vp->bmp);

1169
#if 0
bellard's avatar
bellard committed
1170
    /* XXX: use generic function */
1171
    /* XXX: disable overlay if no hardware acceleration or if RGB format */
1172
    switch(is->video_st->codec->pix_fmt) {
bellard's avatar
bellard committed
1173 1174 1175
    case PIX_FMT_YUV420P:
    case PIX_FMT_YUV422P:
    case PIX_FMT_YUV444P:
1176
    case PIX_FMT_YUYV422:
bellard's avatar
bellard committed
1177 1178 1179 1180 1181 1182 1183 1184 1185
    case PIX_FMT_YUV410P:
    case PIX_FMT_YUV411P:
        is_yuv = 1;
        break;
    default:
        is_yuv = 0;
        break;
    }
#endif
1186 1187
    vp->bmp = SDL_CreateYUVOverlay(is->video_st->codec->width,
                                   is->video_st->codec->height,
1188
                                   SDL_YV12_OVERLAY,
1189
                                   screen);
1190 1191
    vp->width = is->video_st->codec->width;
    vp->height = is->video_st->codec->height;
bellard's avatar
bellard committed
1192 1193 1194 1195 1196 1197 1198

    SDL_LockMutex(is->pictq_mutex);
    vp->allocated = 1;
    SDL_CondSignal(is->pictq_cond);
    SDL_UnlockMutex(is->pictq_mutex);
}

michael's avatar
michael committed
1199 1200 1201 1202
/**
 *
 * @param pts the dts of the pkt / pts of the frame and guessed if not known
 */
1203
static int queue_picture(VideoState *is, AVFrame *src_frame, double pts)
bellard's avatar
bellard committed
1204 1205 1206 1207
{
    VideoPicture *vp;
    int dst_pix_fmt;
    AVPicture pict;
1208
    static struct SwsContext *img_convert_ctx;
1209

bellard's avatar
bellard committed
1210 1211 1212 1213 1214 1215 1216
    /* wait until we have space to put a new picture */
    SDL_LockMutex(is->pictq_mutex);
    while (is->pictq_size >= VIDEO_PICTURE_QUEUE_SIZE &&
           !is->videoq.abort_request) {
        SDL_CondWait(is->pictq_cond, is->pictq_mutex);
    }
    SDL_UnlockMutex(is->pictq_mutex);
1217

bellard's avatar
bellard committed
1218 1219 1220 1221 1222 1223
    if (is->videoq.abort_request)
        return -1;

    vp = &is->pictq[is->pictq_windex];

    /* alloc or resize hardware picture buffer */
1224
    if (!vp->bmp ||
1225 1226
        vp->width != is->video_st->codec->width ||
        vp->height != is->video_st->codec->height) {
bellard's avatar
bellard committed
1227 1228 1229 1230 1231 1232 1233 1234 1235
        SDL_Event event;

        vp->allocated = 0;

        /* the allocation must be done in the main thread to avoid
           locking problems */
        event.type = FF_ALLOC_EVENT;
        event.user.data1 = is;
        SDL_PushEvent(&event);
1236

bellard's avatar
bellard committed
1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247
        /* wait until the picture is allocated */
        SDL_LockMutex(is->pictq_mutex);
        while (!vp->allocated && !is->videoq.abort_request) {
            SDL_CondWait(is->pictq_cond, is->pictq_mutex);
        }
        SDL_UnlockMutex(is->pictq_mutex);

        if (is->videoq.abort_request)
            return -1;
    }

1248
    /* if the frame is not skipped, then display it */
bellard's avatar
bellard committed
1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260
    if (vp->bmp) {
        /* get a pointer on the bitmap */
        SDL_LockYUVOverlay (vp->bmp);

        dst_pix_fmt = PIX_FMT_YUV420P;
        pict.data[0] = vp->bmp->pixels[0];
        pict.data[1] = vp->bmp->pixels[2];
        pict.data[2] = vp->bmp->pixels[1];

        pict.linesize[0] = vp->bmp->pitches[0];
        pict.linesize[1] = vp->bmp->pitches[2];
        pict.linesize[2] = vp->bmp->pitches[1];
1261 1262 1263 1264 1265
        img_convert_ctx = sws_getCachedContext(img_convert_ctx,
            is->video_st->codec->width, is->video_st->codec->height,
            is->video_st->codec->pix_fmt,
            is->video_st->codec->width, is->video_st->codec->height,
            dst_pix_fmt, sws_flags, NULL, NULL, NULL);
alex's avatar
alex committed
1266 1267 1268 1269
        if (img_convert_ctx == NULL) {
            fprintf(stderr, "Cannot initialize the conversion context\n");
            exit(1);
        }
1270 1271
        sws_scale(img_convert_ctx, src_frame->data, src_frame->linesize,
                  0, is->video_st->codec->height, pict.data, pict.linesize);
bellard's avatar
bellard committed
1272 1273 1274
        /* update the bitmap content */
        SDL_UnlockYUVOverlay(vp->bmp);

1275
        vp->pts = pts;
bellard's avatar
bellard committed
1276 1277 1278 1279 1280 1281 1282 1283

        /* now we can update the picture count */
        if (++is->pictq_windex == VIDEO_PICTURE_QUEUE_SIZE)
            is->pictq_windex = 0;
        SDL_LockMutex(is->pictq_mutex);
        is->pictq_size++;
        SDL_UnlockMutex(is->pictq_mutex);
    }
1284 1285 1286
    return 0;
}

1287 1288
/**
 * compute the exact PTS for the picture if it is omitted in the stream
michael's avatar
michael committed
1289 1290
 * @param pts1 the dts of the pkt / pts of the frame
 */
1291 1292 1293
static int output_picture2(VideoState *is, AVFrame *src_frame, double pts1)
{
    double frame_delay, pts;
1294

1295 1296
    pts = pts1;

bellard's avatar
bellard committed
1297
    if (pts != 0) {
1298
        /* update video clock with pts, if present */
bellard's avatar
bellard committed
1299 1300
        is->video_clock = pts;
    } else {
bellard's avatar
bellard committed
1301 1302 1303
        pts = is->video_clock;
    }
    /* update video clock for next frame */
1304
    frame_delay = av_q2d(is->video_st->codec->time_base);
bellard's avatar
bellard committed
1305 1306
    /* for MPEG2, the frame can be repeated, so we update the
       clock accordingly */
michael's avatar
michael committed
1307
    frame_delay += src_frame->repeat_pict * (frame_delay * 0.5);
bellard's avatar
bellard committed
1308
    is->video_clock += frame_delay;
1309 1310 1311 1312 1313 1314 1315 1316 1317 1318

#if defined(DEBUG_SYNC) && 0
    {
        int ftype;
        if (src_frame->pict_type == FF_B_TYPE)
            ftype = 'B';
        else if (src_frame->pict_type == FF_I_TYPE)
            ftype = 'I';
        else
            ftype = 'P';
1319
        printf("frame_type=%c clock=%0.3f pts=%0.3f\n",
bellard's avatar
bellard committed
1320
               ftype, pts, pts1);
1321 1322
    }
#endif
bellard's avatar
bellard committed
1323
    return queue_picture(is, src_frame, pts);
bellard's avatar
bellard committed
1324 1325
}

1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340
static uint64_t global_video_pkt_pts= AV_NOPTS_VALUE;

static int my_get_buffer(struct AVCodecContext *c, AVFrame *pic){
    int ret= avcodec_default_get_buffer(c, pic);
    uint64_t *pts= av_malloc(sizeof(uint64_t));
    *pts= global_video_pkt_pts;
    pic->opaque= pts;
    return ret;
}

static void my_release_buffer(struct AVCodecContext *c, AVFrame *pic){
    if(pic) av_freep(&pic->opaque);
    avcodec_default_release_buffer(c, pic);
}

bellard's avatar
bellard committed
1341 1342 1343 1344
static int video_thread(void *arg)
{
    VideoState *is = arg;
    AVPacket pkt1, *pkt = &pkt1;
bellard's avatar
bellard committed
1345
    int len1, got_picture;
michaelni's avatar
michaelni committed
1346
    AVFrame *frame= avcodec_alloc_frame();
bellard's avatar
bellard committed
1347 1348 1349 1350 1351 1352 1353 1354
    double pts;

    for(;;) {
        while (is->paused && !is->videoq.abort_request) {
            SDL_Delay(10);
        }
        if (packet_queue_get(&is->videoq, pkt, 1) < 0)
            break;
michael's avatar
michael committed
1355 1356 1357 1358 1359 1360

        if(pkt->data == flush_pkt.data){
            avcodec_flush_buffers(is->video_st->codec);
            continue;
        }

1361 1362
        /* NOTE: ipts is the PTS of the _first_ picture beginning in
           this packet, if any */
michael's avatar
michael committed
1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375
        global_video_pkt_pts= pkt->pts;
        len1 = avcodec_decode_video(is->video_st->codec,
                                    frame, &got_picture,
                                    pkt->data, pkt->size);

        if(   (decoder_reorder_pts || pkt->dts == AV_NOPTS_VALUE)
           && frame->opaque && *(uint64_t*)frame->opaque != AV_NOPTS_VALUE)
            pts= *(uint64_t*)frame->opaque;
        else if(pkt->dts != AV_NOPTS_VALUE)
            pts= pkt->dts;
        else
            pts= 0;
        pts *= av_q2d(is->video_st->time_base);
1376

michael's avatar
michael committed
1377 1378
//            if (len1 < 0)
//                break;
michael's avatar
michael committed
1379 1380 1381 1382
        if (got_picture) {
            if (output_picture2(is, frame, pts) < 0)
                goto the_end;
        }
bellard's avatar
bellard committed
1383
        av_free_packet(pkt);
1384
        if (step)
1385 1386
            if (cur_stream)
                stream_pause(cur_stream);
bellard's avatar
bellard committed
1387 1388
    }
 the_end:
michaelni's avatar
michaelni committed
1389
    av_free(frame);
bellard's avatar
bellard committed
1390 1391 1392
    return 0;
}

1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408
static int subtitle_thread(void *arg)
{
    VideoState *is = arg;
    SubPicture *sp;
    AVPacket pkt1, *pkt = &pkt1;
    int len1, got_subtitle;
    double pts;
    int i, j;
    int r, g, b, y, u, v, a;

    for(;;) {
        while (is->paused && !is->subtitleq.abort_request) {
            SDL_Delay(10);
        }
        if (packet_queue_get(&is->subtitleq, pkt, 1) < 0)
            break;
1409

michael's avatar
michael committed
1410 1411 1412 1413
        if(pkt->data == flush_pkt.data){
            avcodec_flush_buffers(is->subtitle_st->codec);
            continue;
        }
1414 1415 1416 1417 1418 1419
        SDL_LockMutex(is->subpq_mutex);
        while (is->subpq_size >= SUBPICTURE_QUEUE_SIZE &&
               !is->subtitleq.abort_request) {
            SDL_CondWait(is->subpq_cond, is->subpq_mutex);
        }
        SDL_UnlockMutex(is->subpq_mutex);
1420

1421 1422
        if (is->subtitleq.abort_request)
            goto the_end;
1423

1424 1425 1426 1427 1428 1429 1430 1431
        sp = &is->subpq[is->subpq_windex];

       /* NOTE: ipts is the PTS of the _first_ picture beginning in
           this packet, if any */
        pts = 0;
        if (pkt->pts != AV_NOPTS_VALUE)
            pts = av_q2d(is->subtitle_st->time_base)*pkt->pts;

1432 1433
        len1 = avcodec_decode_subtitle(is->subtitle_st->codec,
                                    &sp->sub, &got_subtitle,
1434 1435 1436 1437 1438
                                    pkt->data, pkt->size);
//            if (len1 < 0)
//                break;
        if (got_subtitle && sp->sub.format == 0) {
            sp->pts = pts;
1439

1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459
            for (i = 0; i < sp->sub.num_rects; i++)
            {
                for (j = 0; j < sp->sub.rects[i].nb_colors; j++)
                {
                    RGBA_IN(r, g, b, a, sp->sub.rects[i].rgba_palette + j);
                    y = RGB_TO_Y_CCIR(r, g, b);
                    u = RGB_TO_U_CCIR(r, g, b, 0);
                    v = RGB_TO_V_CCIR(r, g, b, 0);
                    YUVA_OUT(sp->sub.rects[i].rgba_palette + j, y, u, v, a);
                }
            }

            /* now we can update the picture count */
            if (++is->subpq_windex == SUBPICTURE_QUEUE_SIZE)
                is->subpq_windex = 0;
            SDL_LockMutex(is->subpq_mutex);
            is->subpq_size++;
            SDL_UnlockMutex(is->subpq_mutex);
        }
        av_free_packet(pkt);
1460
//        if (step)
1461 1462 1463 1464 1465 1466 1467
//            if (cur_stream)
//                stream_pause(cur_stream);
    }
 the_end:
    return 0;
}

bellard's avatar
bellard committed
1468 1469 1470 1471 1472
/* copy samples for viewing in editor window */
static void update_sample_display(VideoState *is, short *samples, int samples_size)
{
    int size, len, channels;

1473
    channels = is->audio_st->codec->channels;
bellard's avatar
bellard committed
1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490

    size = samples_size / sizeof(short);
    while (size > 0) {
        len = SAMPLE_ARRAY_SIZE - is->sample_array_index;
        if (len > size)
            len = size;
        memcpy(is->sample_array + is->sample_array_index, samples, len * sizeof(short));
        samples += len;
        is->sample_array_index += len;
        if (is->sample_array_index >= SAMPLE_ARRAY_SIZE)
            is->sample_array_index = 0;
        size -= len;
    }
}

/* return the new audio buffer size (samples can be added or deleted
   to get better sync if video or external master clock) */
1491
static int synchronize_audio(VideoState *is, short *samples,
1492
                             int samples_size1, double pts)
bellard's avatar
bellard committed
1493
{
1494
    int n, samples_size;
bellard's avatar
bellard committed
1495
    double ref_clock;
1496

1497
    n = 2 * is->audio_st->codec->channels;
1498
    samples_size = samples_size1;
bellard's avatar
bellard committed
1499 1500 1501

    /* if not master, then we try to remove or add samples to correct the clock */
    if (((is->av_sync_type == AV_SYNC_VIDEO_MASTER && is->video_st) ||
1502 1503
         is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
        double diff, avg_diff;
bellard's avatar
bellard committed
1504
        int wanted_size, min_size, max_size, nb_samples;
1505

1506 1507
        ref_clock = get_master_clock(is);
        diff = get_audio_clock(is) - ref_clock;
1508

1509 1510 1511 1512 1513 1514 1515 1516 1517 1518
        if (diff < AV_NOSYNC_THRESHOLD) {
            is->audio_diff_cum = diff + is->audio_diff_avg_coef * is->audio_diff_cum;
            if (is->audio_diff_avg_count < AUDIO_DIFF_AVG_NB) {
                /* not enough measures to have a correct estimate */
                is->audio_diff_avg_count++;
            } else {
                /* estimate the A-V difference */
                avg_diff = is->audio_diff_cum * (1.0 - is->audio_diff_avg_coef);

                if (fabs(avg_diff) >= is->audio_diff_threshold) {
1519
                    wanted_size = samples_size + ((int)(diff * is->audio_st->codec->sample_rate) * n);
1520
                    nb_samples = samples_size / n;
1521

1522 1523 1524 1525 1526 1527
                    min_size = ((nb_samples * (100 - SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
                    max_size = ((nb_samples * (100 + SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
                    if (wanted_size < min_size)
                        wanted_size = min_size;
                    else if (wanted_size > max_size)
                        wanted_size = max_size;
1528

1529 1530 1531 1532 1533 1534 1535
                    /* add or remove samples to correction the synchro */
                    if (wanted_size < samples_size) {
                        /* remove samples */
                        samples_size = wanted_size;
                    } else if (wanted_size > samples_size) {
                        uint8_t *samples_end, *q;
                        int nb;
1536

1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549
                        /* add samples */
                        nb = (samples_size - wanted_size);
                        samples_end = (uint8_t *)samples + samples_size - n;
                        q = samples_end + n;
                        while (nb > 0) {
                            memcpy(q, samples_end, n);
                            q += n;
                            nb -= n;
                        }
                        samples_size = wanted_size;
                    }
                }
#if 0
1550 1551
                printf("diff=%f adiff=%f sample_diff=%d apts=%0.3f vpts=%0.3f %f\n",
                       diff, avg_diff, samples_size - samples_size1,
1552 1553
                       is->audio_clock, is->video_clock, is->audio_diff_threshold);
#endif
bellard's avatar
bellard committed
1554
            }
1555 1556 1557 1558 1559
        } else {
            /* too big difference : may be initial PTS errors, so
               reset A-V filter */
            is->audio_diff_avg_count = 0;
            is->audio_diff_cum = 0;
bellard's avatar
bellard committed
1560 1561 1562 1563 1564 1565 1566
        }
    }

    return samples_size;
}

/* decode one audio frame and returns its uncompressed size */
1567
static int audio_decode_frame(VideoState *is, uint8_t *audio_buf, int buf_size, double *pts_ptr)
bellard's avatar
bellard committed
1568 1569
{
    AVPacket *pkt = &is->audio_pkt;
bellard's avatar
bellard committed
1570
    int n, len1, data_size;
bellard's avatar
bellard committed
1571 1572 1573
    double pts;

    for(;;) {
bellard's avatar
bellard committed
1574
        /* NOTE: the audio packet can contain several frames */
bellard's avatar
bellard committed
1575
        while (is->audio_pkt_size > 0) {
1576 1577
            data_size = buf_size;
            len1 = avcodec_decode_audio2(is->audio_st->codec,
1578
                                        (int16_t *)audio_buf, &data_size,
bellard's avatar
bellard committed
1579
                                        is->audio_pkt_data, is->audio_pkt_size);
bellard's avatar
bellard committed
1580 1581 1582
            if (len1 < 0) {
                /* if error, we skip the frame */
                is->audio_pkt_size = 0;
bellard's avatar
bellard committed
1583
                break;
bellard's avatar
bellard committed
1584
            }
1585

bellard's avatar
bellard committed
1586 1587
            is->audio_pkt_data += len1;
            is->audio_pkt_size -= len1;
bellard's avatar
bellard committed
1588 1589 1590 1591 1592
            if (data_size <= 0)
                continue;
            /* if no pts, then compute it */
            pts = is->audio_clock;
            *pts_ptr = pts;
1593
            n = 2 * is->audio_st->codec->channels;
1594
            is->audio_clock += (double)data_size /
1595
                (double)(n * is->audio_st->codec->sample_rate);
1596
#if defined(DEBUG_SYNC)
bellard's avatar
bellard committed
1597 1598 1599 1600 1601 1602
            {
                static double last_clock;
                printf("audio: delay=%0.3f clock=%0.3f pts=%0.3f\n",
                       is->audio_clock - last_clock,
                       is->audio_clock, pts);
                last_clock = is->audio_clock;
bellard's avatar
bellard committed
1603
            }
bellard's avatar
bellard committed
1604 1605
#endif
            return data_size;
bellard's avatar
bellard committed
1606 1607
        }

bellard's avatar
bellard committed
1608 1609
        /* free the current packet */
        if (pkt->data)
bellard's avatar
bellard committed
1610
            av_free_packet(pkt);
1611

bellard's avatar
bellard committed
1612 1613 1614
        if (is->paused || is->audioq.abort_request) {
            return -1;
        }
1615

bellard's avatar
bellard committed
1616 1617 1618
        /* read next packet */
        if (packet_queue_get(&is->audioq, pkt, 1) < 0)
            return -1;
michael's avatar
michael committed
1619 1620 1621 1622 1623
        if(pkt->data == flush_pkt.data){
            avcodec_flush_buffers(is->audio_st->codec);
            continue;
        }

bellard's avatar
bellard committed
1624 1625
        is->audio_pkt_data = pkt->data;
        is->audio_pkt_size = pkt->size;
1626

bellard's avatar
bellard committed
1627 1628
        /* if update the audio clock with the pts */
        if (pkt->pts != AV_NOPTS_VALUE) {
michael's avatar
michael committed
1629
            is->audio_clock = av_q2d(is->audio_st->time_base)*pkt->pts;
bellard's avatar
bellard committed
1630
        }
bellard's avatar
bellard committed
1631 1632 1633
    }
}

1634 1635 1636
/* get the current audio output buffer size, in samples. With SDL, we
   cannot have a precise information */
static int audio_write_get_buf_size(VideoState *is)
bellard's avatar
bellard committed
1637
{
1638
    return is->audio_buf_size - is->audio_buf_index;
bellard's avatar
bellard committed
1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649
}


/* prepare a new audio buffer */
void sdl_audio_callback(void *opaque, Uint8 *stream, int len)
{
    VideoState *is = opaque;
    int audio_size, len1;
    double pts;

    audio_callback_time = av_gettime();
1650

bellard's avatar
bellard committed
1651 1652
    while (len > 0) {
        if (is->audio_buf_index >= is->audio_buf_size) {
1653
           audio_size = audio_decode_frame(is, is->audio_buf, sizeof(is->audio_buf), &pts);
bellard's avatar
bellard committed
1654 1655 1656 1657 1658 1659 1660
           if (audio_size < 0) {
                /* if error, just output silence */
               is->audio_buf_size = 1024;
               memset(is->audio_buf, 0, is->audio_buf_size);
           } else {
               if (is->show_audio)
                   update_sample_display(is, (int16_t *)is->audio_buf, audio_size);
1661
               audio_size = synchronize_audio(is, (int16_t *)is->audio_buf, audio_size,
bellard's avatar
bellard committed
1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686
                                              pts);
               is->audio_buf_size = audio_size;
           }
           is->audio_buf_index = 0;
        }
        len1 = is->audio_buf_size - is->audio_buf_index;
        if (len1 > len)
            len1 = len;
        memcpy(stream, (uint8_t *)is->audio_buf + is->audio_buf_index, len1);
        len -= len1;
        stream += len1;
        is->audio_buf_index += len1;
    }
}

/* open a given stream. Return 0 if OK */
static int stream_component_open(VideoState *is, int stream_index)
{
    AVFormatContext *ic = is->ic;
    AVCodecContext *enc;
    AVCodec *codec;
    SDL_AudioSpec wanted_spec, spec;

    if (stream_index < 0 || stream_index >= ic->nb_streams)
        return -1;
1687
    enc = ic->streams[stream_index]->codec;
1688

bellard's avatar
bellard committed
1689 1690 1691 1692
    /* prepare audio output */
    if (enc->codec_type == CODEC_TYPE_AUDIO) {
        wanted_spec.freq = enc->sample_rate;
        wanted_spec.format = AUDIO_S16SYS;
1693 1694 1695
        /* hack for AC3. XXX: suppress that */
        if (enc->channels > 2)
            enc->channels = 2;
bellard's avatar
bellard committed
1696 1697
        wanted_spec.channels = enc->channels;
        wanted_spec.silence = 0;
1698
        wanted_spec.samples = SDL_AUDIO_BUFFER_SIZE;
bellard's avatar
bellard committed
1699 1700
        wanted_spec.callback = sdl_audio_callback;
        wanted_spec.userdata = is;
1701 1702
        if (SDL_OpenAudio(&wanted_spec, &spec) < 0) {
            fprintf(stderr, "SDL_OpenAudio: %s\n", SDL_GetError());
bellard's avatar
bellard committed
1703
            return -1;
1704
        }
bellard's avatar
bellard committed
1705 1706 1707 1708
        is->audio_hw_buf_size = spec.size;
    }

    codec = avcodec_find_decoder(enc->codec_id);
michael's avatar
michael committed
1709 1710
    enc->debug_mv = debug_mv;
    enc->debug = debug;
michael's avatar
 
michael committed
1711
    enc->workaround_bugs = workaround_bugs;
michael's avatar
michael committed
1712
    enc->lowres = lowres;
michael's avatar
michael committed
1713
    if(lowres) enc->flags |= CODEC_FLAG_EMU_EDGE;
michael's avatar
michael committed
1714
    enc->idct_algo= idct;
1715
    if(fast) enc->flags2 |= CODEC_FLAG2_FAST;
michael's avatar
michael committed
1716 1717 1718
    enc->skip_frame= skip_frame;
    enc->skip_idct= skip_idct;
    enc->skip_loop_filter= skip_loop_filter;
1719 1720
    enc->error_resilience= error_resilience;
    enc->error_concealment= error_concealment;
bellard's avatar
bellard committed
1721 1722 1723
    if (!codec ||
        avcodec_open(enc, codec) < 0)
        return -1;
michael's avatar
michael committed
1724 1725 1726
    if(thread_count>1)
        avcodec_thread_init(enc, thread_count);
    enc->thread_count= thread_count;
1727
    switch(enc->codec_type) {
bellard's avatar
bellard committed
1728 1729 1730 1731 1732
    case CODEC_TYPE_AUDIO:
        is->audio_stream = stream_index;
        is->audio_st = ic->streams[stream_index];
        is->audio_buf_size = 0;
        is->audio_buf_index = 0;
1733 1734 1735 1736 1737 1738 1739 1740

        /* init averaging filter */
        is->audio_diff_avg_coef = exp(log(0.01) / AUDIO_DIFF_AVG_NB);
        is->audio_diff_avg_count = 0;
        /* since we do not have a precise anough audio fifo fullness,
           we correct audio sync only if larger than this threshold */
        is->audio_diff_threshold = 2.0 * SDL_AUDIO_BUFFER_SIZE / enc->sample_rate;

bellard's avatar
bellard committed
1741 1742
        memset(&is->audio_pkt, 0, sizeof(is->audio_pkt));
        packet_queue_init(&is->audioq);
1743
        SDL_PauseAudio(0);
bellard's avatar
bellard committed
1744 1745 1746 1747 1748
        break;
    case CODEC_TYPE_VIDEO:
        is->video_stream = stream_index;
        is->video_st = ic->streams[stream_index];

1749 1750 1751 1752
        is->frame_last_delay = 40e-3;
        is->frame_timer = (double)av_gettime() / 1000000.0;
        is->video_current_pts_time = av_gettime();

bellard's avatar
bellard committed
1753 1754
        packet_queue_init(&is->videoq);
        is->video_tid = SDL_CreateThread(video_thread, is);
1755 1756 1757

        enc->    get_buffer=     my_get_buffer;
        enc->release_buffer= my_release_buffer;
bellard's avatar
bellard committed
1758
        break;
1759 1760 1761 1762
    case CODEC_TYPE_SUBTITLE:
        is->subtitle_stream = stream_index;
        is->subtitle_st = ic->streams[stream_index];
        packet_queue_init(&is->subtitleq);
1763

1764 1765
        is->subtitle_tid = SDL_CreateThread(subtitle_thread, is);
        break;
bellard's avatar
bellard committed
1766 1767 1768 1769 1770 1771 1772 1773 1774 1775
    default:
        break;
    }
    return 0;
}

static void stream_component_close(VideoState *is, int stream_index)
{
    AVFormatContext *ic = is->ic;
    AVCodecContext *enc;
1776

1777 1778
    if (stream_index < 0 || stream_index >= ic->nb_streams)
        return;
1779
    enc = ic->streams[stream_index]->codec;
bellard's avatar
bellard committed
1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801

    switch(enc->codec_type) {
    case CODEC_TYPE_AUDIO:
        packet_queue_abort(&is->audioq);

        SDL_CloseAudio();

        packet_queue_end(&is->audioq);
        break;
    case CODEC_TYPE_VIDEO:
        packet_queue_abort(&is->videoq);

        /* note: we also signal this mutex to make sure we deblock the
           video thread in all cases */
        SDL_LockMutex(is->pictq_mutex);
        SDL_CondSignal(is->pictq_cond);
        SDL_UnlockMutex(is->pictq_mutex);

        SDL_WaitThread(is->video_tid, NULL);

        packet_queue_end(&is->videoq);
        break;
1802 1803
    case CODEC_TYPE_SUBTITLE:
        packet_queue_abort(&is->subtitleq);
1804

1805 1806 1807 1808
        /* note: we also signal this mutex to make sure we deblock the
           video thread in all cases */
        SDL_LockMutex(is->subpq_mutex);
        is->subtitle_stream_changed = 1;
1809

1810 1811 1812 1813 1814 1815 1816
        SDL_CondSignal(is->subpq_cond);
        SDL_UnlockMutex(is->subpq_mutex);

        SDL_WaitThread(is->subtitle_tid, NULL);

        packet_queue_end(&is->subtitleq);
        break;
bellard's avatar
bellard committed
1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830
    default:
        break;
    }

    avcodec_close(enc);
    switch(enc->codec_type) {
    case CODEC_TYPE_AUDIO:
        is->audio_st = NULL;
        is->audio_stream = -1;
        break;
    case CODEC_TYPE_VIDEO:
        is->video_st = NULL;
        is->video_stream = -1;
        break;
1831 1832 1833 1834
    case CODEC_TYPE_SUBTITLE:
        is->subtitle_st = NULL;
        is->subtitle_stream = -1;
        break;
bellard's avatar
bellard committed
1835 1836 1837 1838 1839
    default:
        break;
    }
}

1840
static void dump_stream_info(const AVFormatContext *s)
bellard's avatar
bellard committed
1841 1842 1843 1844 1845 1846 1847
{
    if (s->track != 0)
        fprintf(stderr, "Track: %d\n", s->track);
    if (s->title[0] != '\0')
        fprintf(stderr, "Title: %s\n", s->title);
    if (s->author[0] != '\0')
        fprintf(stderr, "Author: %s\n", s->author);
gpoirier's avatar
gpoirier committed
1848 1849 1850 1851
    if (s->copyright[0] != '\0')
        fprintf(stderr, "Copyright: %s\n", s->copyright);
    if (s->comment[0] != '\0')
        fprintf(stderr, "Comment: %s\n", s->comment);
bellard's avatar
bellard committed
1852 1853 1854 1855 1856 1857 1858 1859
    if (s->album[0] != '\0')
        fprintf(stderr, "Album: %s\n", s->album);
    if (s->year != 0)
        fprintf(stderr, "Year: %d\n", s->year);
    if (s->genre[0] != '\0')
        fprintf(stderr, "Genre: %s\n", s->genre);
}

1860 1861 1862 1863 1864 1865 1866 1867
/* since we have only one decoding thread, we can use a global
   variable instead of a thread local variable */
static VideoState *global_video_state;

static int decode_interrupt_cb(void)
{
    return (global_video_state && global_video_state->abort_request);
}
bellard's avatar
bellard committed
1868 1869 1870 1871 1872 1873

/* this thread gets the stream from the disk or the network */
static int decode_thread(void *arg)
{
    VideoState *is = arg;
    AVFormatContext *ic;
1874
    int err, i, ret, video_index, audio_index;
bellard's avatar
bellard committed
1875
    AVPacket pkt1, *pkt = &pkt1;
1876
    AVFormatParameters params, *ap = &params;
bellard's avatar
bellard committed
1877 1878 1879 1880 1881

    video_index = -1;
    audio_index = -1;
    is->video_stream = -1;
    is->audio_stream = -1;
1882
    is->subtitle_stream = -1;
bellard's avatar
bellard committed
1883

1884 1885 1886
    global_video_state = is;
    url_set_interrupt_cb(decode_interrupt_cb);

1887
    memset(ap, 0, sizeof(*ap));
1888

1889 1890
    ap->width = frame_width;
    ap->height= frame_height;
michael's avatar
michael committed
1891
    ap->time_base= (AVRational){1, 25};
1892
    ap->pix_fmt = frame_pix_fmt;
michael's avatar
michael committed
1893

1894
    err = av_open_input_file(&ic, is->filename, is->iformat, 0, ap);
1895 1896 1897 1898 1899
    if (err < 0) {
        print_error(is->filename, err);
        ret = -1;
        goto fail;
    }
bellard's avatar
bellard committed
1900
    is->ic = ic;
1901 1902 1903 1904

    if(genpts)
        ic->flags |= AVFMT_FLAG_GENPTS;

1905 1906 1907 1908 1909 1910 1911
    err = av_find_stream_info(ic);
    if (err < 0) {
        fprintf(stderr, "%s: could not find codec parameters\n", is->filename);
        ret = -1;
        goto fail;
    }
    ic->pb.eof_reached= 0; //FIXME hack, ffplay maybe should not use url_feof() to test for the end
bellard's avatar
bellard committed
1912 1913 1914 1915 1916 1917 1918 1919 1920

    /* if seeking requested, we execute it */
    if (start_time != AV_NOPTS_VALUE) {
        int64_t timestamp;

        timestamp = start_time;
        /* add the stream start time */
        if (ic->start_time != AV_NOPTS_VALUE)
            timestamp += ic->start_time;
michael's avatar
michael committed
1921
        ret = av_seek_frame(ic, -1, timestamp, AVSEEK_FLAG_BACKWARD);
bellard's avatar
bellard committed
1922
        if (ret < 0) {
1923
            fprintf(stderr, "%s: could not seek to position %0.3f\n",
bellard's avatar
bellard committed
1924 1925 1926 1927
                    is->filename, (double)timestamp / AV_TIME_BASE);
        }
    }

bellard's avatar
bellard committed
1928
    for(i = 0; i < ic->nb_streams; i++) {
1929
        AVCodecContext *enc = ic->streams[i]->codec;
bellard's avatar
bellard committed
1930 1931
        switch(enc->codec_type) {
        case CODEC_TYPE_AUDIO:
1932
            if ((audio_index < 0 || wanted_audio_stream-- > 0) && !audio_disable)
bellard's avatar
bellard committed
1933 1934 1935
                audio_index = i;
            break;
        case CODEC_TYPE_VIDEO:
1936
            if ((video_index < 0 || wanted_video_stream-- > 0) && !video_disable)
bellard's avatar
bellard committed
1937 1938 1939 1940 1941 1942 1943 1944
                video_index = i;
            break;
        default:
            break;
        }
    }
    if (show_status) {
        dump_format(ic, 0, is->filename, 0);
bellard's avatar
bellard committed
1945
        dump_stream_info(ic);
bellard's avatar
bellard committed
1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960
    }

    /* open the streams */
    if (audio_index >= 0) {
        stream_component_open(is, audio_index);
    }

    if (video_index >= 0) {
        stream_component_open(is, video_index);
    } else {
        if (!display_disable)
            is->show_audio = 1;
    }

    if (is->video_stream < 0 && is->audio_stream < 0) {
1961 1962
        fprintf(stderr, "%s: could not open codecs\n", is->filename);
        ret = -1;
bellard's avatar
bellard committed
1963 1964 1965 1966 1967 1968
        goto fail;
    }

    for(;;) {
        if (is->abort_request)
            break;
1969 1970
        if (is->paused != is->last_paused) {
            is->last_paused = is->paused;
bellard's avatar
bellard committed
1971 1972 1973 1974
            if (is->paused)
                av_read_pause(ic);
            else
                av_read_play(ic);
1975
        }
1976
#ifdef CONFIG_RTSP_DEMUXER
1977
        if (is->paused && ic->iformat == &rtsp_demuxer) {
1978 1979 1980 1981 1982
            /* wait 10 ms to avoid trying to get another packet */
            /* XXX: horrible */
            SDL_Delay(10);
            continue;
        }
1983
#endif
bellard's avatar
bellard committed
1984
        if (is->seek_req) {
1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996
            int stream_index= -1;
            int64_t seek_target= is->seek_pos;

            if     (is->   video_stream >= 0) stream_index= is->   video_stream;
            else if(is->   audio_stream >= 0) stream_index= is->   audio_stream;
            else if(is->subtitle_stream >= 0) stream_index= is->subtitle_stream;

            if(stream_index>=0){
                seek_target= av_rescale_q(seek_target, AV_TIME_BASE_Q, ic->streams[stream_index]->time_base);
            }

            ret = av_seek_frame(is->ic, stream_index, seek_target, is->seek_flags);
bellard's avatar
bellard committed
1997 1998
            if (ret < 0) {
                fprintf(stderr, "%s: error while seeking\n", is->ic->filename);
michael's avatar
michael committed
1999 2000 2001
            }else{
                if (is->audio_stream >= 0) {
                    packet_queue_flush(&is->audioq);
michael's avatar
michael committed
2002
                    packet_queue_put(&is->audioq, &flush_pkt);
michael's avatar
michael committed
2003
                }
2004 2005
                if (is->subtitle_stream >= 0) {
                    packet_queue_flush(&is->subtitleq);
michael's avatar
michael committed
2006
                    packet_queue_put(&is->subtitleq, &flush_pkt);
2007
                }
michael's avatar
michael committed
2008 2009
                if (is->video_stream >= 0) {
                    packet_queue_flush(&is->videoq);
michael's avatar
michael committed
2010
                    packet_queue_put(&is->videoq, &flush_pkt);
michael's avatar
michael committed
2011
                }
bellard's avatar
bellard committed
2012 2013 2014
            }
            is->seek_req = 0;
        }
2015

bellard's avatar
bellard committed
2016 2017
        /* if the queue are full, no need to read more */
        if (is->audioq.size > MAX_AUDIOQ_SIZE ||
2018 2019
            is->videoq.size > MAX_VIDEOQ_SIZE ||
            is->subtitleq.size > MAX_SUBTITLEQ_SIZE ||
2020
            url_feof(&ic->pb)) {
bellard's avatar
bellard committed
2021 2022 2023 2024
            /* wait 10 ms */
            SDL_Delay(10);
            continue;
        }
bellard's avatar
bellard committed
2025
        ret = av_read_frame(ic, pkt);
bellard's avatar
bellard committed
2026
        if (ret < 0) {
2027
            if (url_ferror(&ic->pb) == 0) {
romansh's avatar
 
romansh committed
2028
                SDL_Delay(100); /* wait for user event */
2029 2030 2031
                continue;
            } else
                break;
bellard's avatar
bellard committed
2032 2033 2034 2035 2036
        }
        if (pkt->stream_index == is->audio_stream) {
            packet_queue_put(&is->audioq, pkt);
        } else if (pkt->stream_index == is->video_stream) {
            packet_queue_put(&is->videoq, pkt);
2037 2038
        } else if (pkt->stream_index == is->subtitle_stream) {
            packet_queue_put(&is->subtitleq, pkt);
bellard's avatar
bellard committed
2039 2040 2041 2042 2043 2044 2045 2046 2047
        } else {
            av_free_packet(pkt);
        }
    }
    /* wait until the end */
    while (!is->abort_request) {
        SDL_Delay(100);
    }

2048
    ret = 0;
bellard's avatar
bellard committed
2049
 fail:
2050 2051 2052
    /* disable interrupting */
    global_video_state = NULL;

bellard's avatar
bellard committed
2053 2054 2055 2056 2057
    /* close each stream */
    if (is->audio_stream >= 0)
        stream_component_close(is, is->audio_stream);
    if (is->video_stream >= 0)
        stream_component_close(is, is->video_stream);
2058 2059
    if (is->subtitle_stream >= 0)
        stream_component_close(is, is->subtitle_stream);
2060 2061 2062 2063
    if (is->ic) {
        av_close_input_file(is->ic);
        is->ic = NULL; /* safety */
    }
2064 2065
    url_set_interrupt_cb(NULL);

2066 2067
    if (ret != 0) {
        SDL_Event event;
2068

2069 2070 2071 2072
        event.type = FF_QUIT_EVENT;
        event.user.data1 = is;
        SDL_PushEvent(&event);
    }
bellard's avatar
bellard committed
2073 2074 2075
    return 0;
}

2076
static VideoState *stream_open(const char *filename, AVInputFormat *iformat)
bellard's avatar
bellard committed
2077 2078 2079 2080 2081 2082
{
    VideoState *is;

    is = av_mallocz(sizeof(VideoState));
    if (!is)
        return NULL;
2083
    av_strlcpy(is->filename, filename, sizeof(is->filename));
2084
    is->iformat = iformat;
bellard's avatar
bellard committed
2085 2086 2087 2088 2089 2090
    is->ytop = 0;
    is->xleft = 0;

    /* start video display */
    is->pictq_mutex = SDL_CreateMutex();
    is->pictq_cond = SDL_CreateCond();
2091

2092 2093
    is->subpq_mutex = SDL_CreateMutex();
    is->subpq_cond = SDL_CreateCond();
2094

bellard's avatar
bellard committed
2095 2096 2097
    /* add the refresh timer to draw the picture */
    schedule_refresh(is, 40);

2098
    is->av_sync_type = av_sync_type;
bellard's avatar
bellard committed
2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114 2115 2116 2117 2118 2119 2120 2121 2122 2123 2124
    is->parse_tid = SDL_CreateThread(decode_thread, is);
    if (!is->parse_tid) {
        av_free(is);
        return NULL;
    }
    return is;
}

static void stream_close(VideoState *is)
{
    VideoPicture *vp;
    int i;
    /* XXX: use a special url_shutdown call to abort parse cleanly */
    is->abort_request = 1;
    SDL_WaitThread(is->parse_tid, NULL);

    /* free all pictures */
    for(i=0;i<VIDEO_PICTURE_QUEUE_SIZE; i++) {
        vp = &is->pictq[i];
        if (vp->bmp) {
            SDL_FreeYUVOverlay(vp->bmp);
            vp->bmp = NULL;
        }
    }
    SDL_DestroyMutex(is->pictq_mutex);
    SDL_DestroyCond(is->pictq_cond);
2125 2126
    SDL_DestroyMutex(is->subpq_mutex);
    SDL_DestroyCond(is->subpq_cond);
bellard's avatar
bellard committed
2127 2128
}

2129
static void stream_cycle_channel(VideoState *is, int codec_type)
2130 2131 2132 2133 2134 2135 2136
{
    AVFormatContext *ic = is->ic;
    int start_index, stream_index;
    AVStream *st;

    if (codec_type == CODEC_TYPE_VIDEO)
        start_index = is->video_stream;
2137
    else if (codec_type == CODEC_TYPE_AUDIO)
2138
        start_index = is->audio_stream;
2139 2140 2141
    else
        start_index = is->subtitle_stream;
    if (start_index < (codec_type == CODEC_TYPE_SUBTITLE ? -1 : 0))
2142 2143 2144 2145
        return;
    stream_index = start_index;
    for(;;) {
        if (++stream_index >= is->ic->nb_streams)
2146 2147 2148 2149 2150 2151 2152 2153
        {
            if (codec_type == CODEC_TYPE_SUBTITLE)
            {
                stream_index = -1;
                goto the_end;
            } else
                stream_index = 0;
        }
2154 2155 2156
        if (stream_index == start_index)
            return;
        st = ic->streams[stream_index];
2157
        if (st->codec->codec_type == codec_type) {
2158 2159 2160
            /* check that parameters are OK */
            switch(codec_type) {
            case CODEC_TYPE_AUDIO:
2161 2162
                if (st->codec->sample_rate != 0 &&
                    st->codec->channels != 0)
2163 2164 2165
                    goto the_end;
                break;
            case CODEC_TYPE_VIDEO:
2166
            case CODEC_TYPE_SUBTITLE:
2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178
                goto the_end;
            default:
                break;
            }
        }
    }
 the_end:
    stream_component_close(is, start_index);
    stream_component_open(is, stream_index);
}


2179
static void toggle_full_screen(void)
bellard's avatar
bellard committed
2180 2181
{
    is_full_screen = !is_full_screen;
2182 2183
    if (!fs_screen_width) {
        /* use default SDL method */
2184
//        SDL_WM_ToggleFullScreen(screen);
bellard's avatar
bellard committed
2185
    }
2186
    video_open(cur_stream);
bellard's avatar
bellard committed
2187 2188
}

2189
static void toggle_pause(void)
bellard's avatar
bellard committed
2190 2191 2192
{
    if (cur_stream)
        stream_pause(cur_stream);
2193 2194 2195
    step = 0;
}

2196
static void step_to_next_frame(void)
2197 2198
{
    if (cur_stream) {
2199
        /* if the stream is paused unpause it, then step */
2200
        if (cur_stream->paused)
2201
            stream_pause(cur_stream);
2202 2203
    }
    step = 1;
bellard's avatar
bellard committed
2204 2205
}

2206
static void do_exit(void)
bellard's avatar
bellard committed
2207 2208 2209 2210 2211 2212 2213 2214 2215 2216 2217
{
    if (cur_stream) {
        stream_close(cur_stream);
        cur_stream = NULL;
    }
    if (show_status)
        printf("\n");
    SDL_Quit();
    exit(0);
}

2218
static void toggle_audio_display(void)
bellard's avatar
bellard committed
2219 2220 2221 2222 2223 2224 2225
{
    if (cur_stream) {
        cur_stream->show_audio = !cur_stream->show_audio;
    }
}

/* handle an event sent by the GUI */
2226
static void event_loop(void)
bellard's avatar
bellard committed
2227 2228
{
    SDL_Event event;
2229
    double incr, pos, frac;
bellard's avatar
bellard committed
2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246

    for(;;) {
        SDL_WaitEvent(&event);
        switch(event.type) {
        case SDL_KEYDOWN:
            switch(event.key.keysym.sym) {
            case SDLK_ESCAPE:
            case SDLK_q:
                do_exit();
                break;
            case SDLK_f:
                toggle_full_screen();
                break;
            case SDLK_p:
            case SDLK_SPACE:
                toggle_pause();
                break;
2247 2248 2249
            case SDLK_s: //S: Step to next frame
                step_to_next_frame();
                break;
bellard's avatar
bellard committed
2250
            case SDLK_a:
2251
                if (cur_stream)
2252 2253 2254
                    stream_cycle_channel(cur_stream, CODEC_TYPE_AUDIO);
                break;
            case SDLK_v:
2255
                if (cur_stream)
2256 2257
                    stream_cycle_channel(cur_stream, CODEC_TYPE_VIDEO);
                break;
2258
            case SDLK_t:
2259
                if (cur_stream)
2260 2261
                    stream_cycle_channel(cur_stream, CODEC_TYPE_SUBTITLE);
                break;
2262
            case SDLK_w:
bellard's avatar
bellard committed
2263 2264
                toggle_audio_display();
                break;
bellard's avatar
bellard committed
2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277
            case SDLK_LEFT:
                incr = -10.0;
                goto do_seek;
            case SDLK_RIGHT:
                incr = 10.0;
                goto do_seek;
            case SDLK_UP:
                incr = 60.0;
                goto do_seek;
            case SDLK_DOWN:
                incr = -60.0;
            do_seek:
                if (cur_stream) {
2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288 2289 2290
                    if (seek_by_bytes) {
                        pos = url_ftell(&cur_stream->ic->pb);
                        if (cur_stream->ic->bit_rate)
                            incr *= cur_stream->ic->bit_rate / 60.0;
                        else
                            incr *= 180000.0;
                        pos += incr;
                        stream_seek(cur_stream, pos, incr);
                    } else {
                        pos = get_master_clock(cur_stream);
                        pos += incr;
                        stream_seek(cur_stream, (int64_t)(pos * AV_TIME_BASE), incr);
                    }
bellard's avatar
bellard committed
2291 2292
                }
                break;
bellard's avatar
bellard committed
2293 2294 2295 2296
            default:
                break;
            }
            break;
2297
        case SDL_MOUSEBUTTONDOWN:
2298 2299 2300 2301 2302 2303 2304 2305 2306 2307 2308 2309 2310 2311 2312 2313 2314
            if (cur_stream) {
                int ns, hh, mm, ss;
                int tns, thh, tmm, tss;
                tns = cur_stream->ic->duration/1000000LL;
                thh = tns/3600;
                tmm = (tns%3600)/60;
                tss = (tns%60);
                frac = (double)event.button.x/(double)cur_stream->width;
                ns = frac*tns;
                hh = ns/3600;
                mm = (ns%3600)/60;
                ss = (ns%60);
                fprintf(stderr, "Seek to %2.0f%% (%2d:%02d:%02d) of total duration (%2d:%02d:%02d)       \n", frac*100,
                        hh, mm, ss, thh, tmm, tss);
                stream_seek(cur_stream, (int64_t)(cur_stream->ic->start_time+frac*cur_stream->ic->duration), 0);
            }
            break;
bellard's avatar
bellard committed
2315 2316
        case SDL_VIDEORESIZE:
            if (cur_stream) {
2317
                screen = SDL_SetVideoMode(event.resize.w, event.resize.h, 0,
bellard's avatar
bellard committed
2318
                                          SDL_HWSURFACE|SDL_RESIZABLE|SDL_ASYNCBLIT|SDL_HWACCEL);
2319 2320
                screen_width = cur_stream->width = event.resize.w;
                screen_height= cur_stream->height= event.resize.h;
bellard's avatar
bellard committed
2321 2322 2323
            }
            break;
        case SDL_QUIT:
2324
        case FF_QUIT_EVENT:
bellard's avatar
bellard committed
2325 2326 2327
            do_exit();
            break;
        case FF_ALLOC_EVENT:
michael's avatar
michael committed
2328
            video_open(event.user.data1);
bellard's avatar
bellard committed
2329 2330 2331 2332 2333 2334 2335 2336 2337 2338 2339
            alloc_picture(event.user.data1);
            break;
        case FF_REFRESH_EVENT:
            video_refresh_timer(event.user.data1);
            break;
        default:
            break;
        }
    }
}

2340 2341
static void opt_frame_size(const char *arg)
{
2342
    if (av_parse_video_frame_size(&frame_width, &frame_height, arg) < 0) {
2343 2344 2345 2346 2347 2348 2349 2350 2351
        fprintf(stderr, "Incorrect frame size\n");
        exit(1);
    }
    if ((frame_width % 2) != 0 || (frame_height % 2) != 0) {
        fprintf(stderr, "Frame size must be a multiple of 2\n");
        exit(1);
    }
}

2352
static void opt_width(const char *arg)
bellard's avatar
bellard committed
2353 2354
{
    screen_width = atoi(arg);
michael's avatar
michael committed
2355 2356 2357 2358
    if(screen_width<=0){
        fprintf(stderr, "invalid width\n");
        exit(1);
    }
bellard's avatar
bellard committed
2359 2360
}

2361
static void opt_height(const char *arg)
bellard's avatar
bellard committed
2362 2363
{
    screen_height = atoi(arg);
michael's avatar
michael committed
2364 2365 2366 2367
    if(screen_height<=0){
        fprintf(stderr, "invalid height\n");
        exit(1);
    }
bellard's avatar
bellard committed
2368 2369 2370 2371 2372 2373 2374 2375 2376 2377
}

static void opt_format(const char *arg)
{
    file_iformat = av_find_input_format(arg);
    if (!file_iformat) {
        fprintf(stderr, "Unknown input format: %s\n", arg);
        exit(1);
    }
}
2378

2379 2380 2381 2382 2383
static void opt_frame_pix_fmt(const char *arg)
{
    frame_pix_fmt = avcodec_get_pix_fmt(arg);
}

2384
#ifdef CONFIG_RTSP_DEMUXER
2385
static void opt_rtp_tcp(void)
2386 2387 2388 2389
{
    /* only tcp protocol */
    rtsp_default_protocols = (1 << RTSP_PROTOCOL_RTP_TCP);
}
2390
#endif
2391

2392
static void opt_sync(const char *arg)
2393 2394 2395 2396 2397 2398 2399
{
    if (!strcmp(arg, "audio"))
        av_sync_type = AV_SYNC_AUDIO_MASTER;
    else if (!strcmp(arg, "video"))
        av_sync_type = AV_SYNC_VIDEO_MASTER;
    else if (!strcmp(arg, "ext"))
        av_sync_type = AV_SYNC_EXTERNAL_CLOCK;
2400
    else {
2401
        show_help();
2402 2403
        exit(1);
    }
2404 2405
}

2406
static void opt_seek(const char *arg)
bellard's avatar
bellard committed
2407 2408 2409 2410
{
    start_time = parse_date(arg, 1);
}

2411 2412
static void opt_debug(const char *arg)
{
2413
    av_log_level = 99;
2414 2415
    debug = atoi(arg);
}
2416

2417 2418 2419 2420
static void opt_vismv(const char *arg)
{
    debug_mv = atoi(arg);
}
michael's avatar
michael committed
2421 2422 2423 2424

static void opt_thread_count(const char *arg)
{
    thread_count= atoi(arg);
2425
#if !defined(HAVE_THREADS)
michael's avatar
michael committed
2426 2427 2428
    fprintf(stderr, "Warning: not compiled with thread support, using thread emulation\n");
#endif
}
2429

2430 2431 2432 2433 2434 2435
static void opt_show_help(void)
{
    show_help();
    exit(0);
}

bellard's avatar
bellard committed
2436
const OptionDef options[] = {
2437
    { "h", 0, {(void*)opt_show_help}, "show help" },
bellard's avatar
bellard committed
2438 2439
    { "x", HAS_ARG, {(void*)opt_width}, "force displayed width", "width" },
    { "y", HAS_ARG, {(void*)opt_height}, "force displayed height", "height" },
2440
    { "s", HAS_ARG | OPT_VIDEO, {(void*)opt_frame_size}, "set frame size (WxH or abbreviation)", "size" },
2441
    { "fs", OPT_BOOL, {(void*)&is_full_screen}, "force full screen" },
bellard's avatar
bellard committed
2442 2443
    { "an", OPT_BOOL, {(void*)&audio_disable}, "disable audio" },
    { "vn", OPT_BOOL, {(void*)&video_disable}, "disable video" },
2444
    { "ast", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_audio_stream}, "", "" },
2445
    { "vst", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_video_stream}, "", "" },
bellard's avatar
bellard committed
2446
    { "ss", HAS_ARG, {(void*)&opt_seek}, "seek to a given position in seconds", "pos" },
2447
    { "bytes", OPT_BOOL, {(void*)&seek_by_bytes}, "seek by bytes" },
bellard's avatar
bellard committed
2448 2449
    { "nodisp", OPT_BOOL, {(void*)&display_disable}, "disable graphical display" },
    { "f", HAS_ARG, {(void*)opt_format}, "force format", "fmt" },
2450
    { "pix_fmt", HAS_ARG | OPT_EXPERT | OPT_VIDEO, {(void*)opt_frame_pix_fmt}, "set pixel format", "format" },
bellard's avatar
bellard committed
2451
    { "stats", OPT_BOOL | OPT_EXPERT, {(void*)&show_status}, "show status", "" },
2452
    { "debug", HAS_ARG | OPT_EXPERT, {(void*)opt_debug}, "print specific debug info", "" },
michael's avatar
 
michael committed
2453
    { "bug", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&workaround_bugs}, "workaround bugs", "" },
2454
    { "vismv", HAS_ARG | OPT_EXPERT, {(void*)opt_vismv}, "visualize motion vectors", "" },
2455
    { "fast", OPT_BOOL | OPT_EXPERT, {(void*)&fast}, "non spec compliant optimizations", "" },
2456
    { "genpts", OPT_BOOL | OPT_EXPERT, {(void*)&genpts}, "generate pts", "" },
2457
    { "drp", OPT_BOOL |OPT_EXPERT, {(void*)&decoder_reorder_pts}, "let decoder reorder pts", ""},
michael's avatar
michael committed
2458
    { "lowres", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&lowres}, "", "" },
michael's avatar
michael committed
2459 2460 2461
    { "skiploop", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_loop_filter}, "", "" },
    { "skipframe", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_frame}, "", "" },
    { "skipidct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_idct}, "", "" },
michael's avatar
michael committed
2462
    { "idct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&idct}, "set idct algo",  "algo" },
2463 2464
    { "er", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_resilience}, "set error detection threshold (0-4)",  "threshold" },
    { "ec", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_concealment}, "set error concealment options",  "bit_mask" },
2465
#ifdef CONFIG_RTSP_DEMUXER
2466
    { "rtp_tcp", OPT_EXPERT, {(void*)&opt_rtp_tcp}, "force RTP/TCP protocol usage", "" },
2467
#endif
michael's avatar
michael committed
2468 2469
    { "sync", HAS_ARG | OPT_EXPERT, {(void*)opt_sync}, "set audio-video sync. type (type=audio/video/ext)", "type" },
    { "threads", HAS_ARG | OPT_EXPERT, {(void*)opt_thread_count}, "thread count", "count" },
bellard's avatar
bellard committed
2470 2471 2472 2473 2474
    { NULL, },
};

void show_help(void)
{
benoit's avatar
benoit committed
2475
    printf("ffplay version " FFMPEG_VERSION ", Copyright (c) 2003-2007 Fabrice Bellard, et al.\n"
2476
           "usage: ffplay [options] input_file\n"
bellard's avatar
bellard committed
2477 2478
           "Simple media player\n");
    printf("\n");
2479 2480 2481 2482
    show_help_options(options, "Main options:\n",
                      OPT_EXPERT, 0);
    show_help_options(options, "\nAdvanced options:\n",
                      OPT_EXPERT, OPT_EXPERT);
bellard's avatar
bellard committed
2483 2484 2485 2486
    printf("\nWhile playing:\n"
           "q, ESC              quit\n"
           "f                   toggle full screen\n"
           "p, SPC              pause\n"
2487 2488
           "a                   cycle audio channel\n"
           "v                   cycle video channel\n"
2489
           "t                   cycle subtitle channel\n"
2490
           "w                   show audio waves\n"
bellard's avatar
bellard committed
2491 2492
           "left/right          seek backward/forward 10 seconds\n"
           "down/up             seek backward/forward 1 minute\n"
2493
           "mouse click         seek to percentage in file corresponding to fraction of width\n"
bellard's avatar
bellard committed
2494 2495 2496
           );
}

2497
void opt_input_file(const char *filename)
bellard's avatar
bellard committed
2498
{
2499
    if (!strcmp(filename, "-"))
diego's avatar
diego committed
2500
        filename = "pipe:";
bellard's avatar
bellard committed
2501 2502 2503 2504 2505 2506
    input_filename = filename;
}

/* Called from the main */
int main(int argc, char **argv)
{
michael's avatar
michael committed
2507
    int flags;
2508

bellard's avatar
bellard committed
2509 2510 2511
    /* register all codecs, demux and protocols */
    av_register_all();

2512
    parse_options(argc, argv, options, opt_input_file);
bellard's avatar
bellard committed
2513

2514
    if (!input_filename) {
bellard's avatar
bellard committed
2515
        show_help();
2516 2517
        exit(1);
    }
bellard's avatar
bellard committed
2518 2519 2520 2521

    if (display_disable) {
        video_disable = 1;
    }
2522
    flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER;
2523 2524
#if !defined(__MINGW32__) && !defined(__APPLE__)
    flags |= SDL_INIT_EVENTTHREAD; /* Not supported on Windows or Mac OS X */
2525
#endif
bellard's avatar
bellard committed
2526
    if (SDL_Init (flags)) {
2527
        fprintf(stderr, "Could not initialize SDL - %s\n", SDL_GetError());
bellard's avatar
bellard committed
2528 2529 2530 2531
        exit(1);
    }

    if (!display_disable) {
2532
#ifdef HAVE_SDL_VIDEO_SIZE
2533 2534 2535
        const SDL_VideoInfo *vi = SDL_GetVideoInfo();
        fs_screen_width = vi->current_w;
        fs_screen_height = vi->current_h;
2536
#endif
bellard's avatar
bellard committed
2537 2538 2539 2540 2541 2542 2543
    }

    SDL_EventState(SDL_ACTIVEEVENT, SDL_IGNORE);
    SDL_EventState(SDL_MOUSEMOTION, SDL_IGNORE);
    SDL_EventState(SDL_SYSWMEVENT, SDL_IGNORE);
    SDL_EventState(SDL_USEREVENT, SDL_IGNORE);

michael's avatar
michael committed
2544 2545 2546
    av_init_packet(&flush_pkt);
    flush_pkt.data= "FLUSH";

2547
    cur_stream = stream_open(input_filename, file_iformat);
bellard's avatar
bellard committed
2548 2549 2550 2551 2552 2553 2554

    event_loop();

    /* never returns */

    return 0;
}