|
|
|
|
#include "./libavformat/avformat.h"
|
|
|
|
|
|
|
|
|
|
#if defined(CONFIG_WIN32)
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <sys/timeb.h>
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
#else
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <SDL.h>
|
|
|
|
|
#include <SDL_thread.h>
|
|
|
|
|
|
|
|
|
|
#ifdef CONFIG_WIN32
|
|
|
|
|
#undef main // We don't want SDL to override our main()
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#pragma comment(lib, "SDL.lib")
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD>˳<EFBFBD><EFBFBD>¼<EFBFBD> */
|
|
|
|
|
#define FF_QUIT_EVENT (SDL_USEREVENT + 2)
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD>е<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
#define MAX_VIDEOQ_SIZE (5 * 256 * 1024)
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD>е<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
#define MAX_AUDIOQ_SIZE (5 * 16 * 1024)
|
|
|
|
|
|
|
|
|
|
/* ͼ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>еij<EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
#define VIDEO_PICTURE_QUEUE_SIZE 1
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
typedef struct PacketQueue
|
|
|
|
|
{
|
|
|
|
|
// ͷָ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>βָ<EFBFBD><EFBFBD>
|
|
|
|
|
AVPacketList *first_pkt, *last_pkt;
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD>г<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
int size;
|
|
|
|
|
// <EFBFBD>ж<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
int abort_request;
|
|
|
|
|
// <EFBFBD><EFBFBD>
|
|
|
|
|
SDL_mutex *mutex;
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
SDL_cond *cond;
|
|
|
|
|
} PacketQueue;
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶͼ<EFBFBD><EFBFBD> */
|
|
|
|
|
typedef struct VideoPicture
|
|
|
|
|
{
|
|
|
|
|
// <EFBFBD><EFBFBD>ʾ<EFBFBD><EFBFBD>
|
|
|
|
|
SDL_Overlay *bmp;
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
int width, height; // source height & width
|
|
|
|
|
} VideoPicture;
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>״̬ */
|
|
|
|
|
typedef struct VideoState
|
|
|
|
|
{
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>
|
|
|
|
|
SDL_Thread *parse_tid;
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>
|
|
|
|
|
SDL_Thread *video_tid;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD>ж<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
int abort_request;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>ʾһ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ģ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>߱<EFBFBD>ʾһ<EFBFBD><EFBFBD><EFBFBD>ļ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ģ<EFBFBD>
|
|
|
|
|
AVFormatContext *ic;
|
|
|
|
|
|
|
|
|
|
int audio_stream;
|
|
|
|
|
int video_stream;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>
|
|
|
|
|
AVStream *audio_st;
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>
|
|
|
|
|
AVStream *video_st;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
PacketQueue audioq;
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
PacketQueue videoq;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD>Ŷ<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
VideoPicture pictq[VIDEO_PICTURE_QUEUE_SIZE];
|
|
|
|
|
double frame_last_delay;
|
|
|
|
|
|
|
|
|
|
uint8_t audio_buf[(AVCODEC_MAX_AUDIO_FRAME_SIZE *3) / 2];
|
|
|
|
|
unsigned int audio_buf_size;
|
|
|
|
|
int audio_buf_index;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD>
|
|
|
|
|
AVPacket audio_pkt;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD>ݺͳ<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
uint8_t *audio_pkt_data;
|
|
|
|
|
int audio_pkt_size;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
SDL_mutex *video_decoder_mutex;
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
SDL_mutex *audio_decoder_mutex;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD>ļ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
char filename[240];
|
|
|
|
|
|
|
|
|
|
} VideoState;
|
|
|
|
|
|
|
|
|
|
static AVInputFormat *file_iformat;
|
|
|
|
|
static const char *input_filename;
|
|
|
|
|
static VideoState *cur_stream;
|
|
|
|
|
|
|
|
|
|
static SDL_Surface *screen;
|
|
|
|
|
/* <EFBFBD><EFBFBD>ȡ<EFBFBD><EFBFBD>ǰ<EFBFBD><EFBFBD>ʱ<EFBFBD><EFBFBD> */
|
|
|
|
|
int64_t av_gettime(void)
|
|
|
|
|
{
|
|
|
|
|
#if defined(CONFIG_WINCE)
|
|
|
|
|
return timeGetTime() *int64_t_C(1000);
|
|
|
|
|
#elif defined(CONFIG_WIN32)
|
|
|
|
|
struct _timeb tb;
|
|
|
|
|
_ftime(&tb);
|
|
|
|
|
return ((int64_t)tb.time *int64_t_C(1000) + (int64_t)tb.millitm) *int64_t_C(1000);
|
|
|
|
|
#else
|
|
|
|
|
struct timeval tv;
|
|
|
|
|
gettimeofday(&tv, NULL);
|
|
|
|
|
return (int64_t)tv.tv_sec *1000000+tv.tv_usec;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>г<EFBFBD>ʼ<EFBFBD><EFBFBD> */
|
|
|
|
|
static void packet_queue_init(PacketQueue *q) // packet queue handling
|
|
|
|
|
{
|
|
|
|
|
memset(q, 0, sizeof(PacketQueue));
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
q->mutex = SDL_CreateMutex();
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
q->cond = SDL_CreateCond();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ˢ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>У<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ͷ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD> */
|
|
|
|
|
static void packet_queue_flush(PacketQueue *q)
|
|
|
|
|
{
|
|
|
|
|
AVPacketList *pkt, *pkt1;
|
|
|
|
|
|
|
|
|
|
SDL_LockMutex(q->mutex);
|
|
|
|
|
for (pkt = q->first_pkt; pkt != NULL; pkt = pkt1)
|
|
|
|
|
{
|
|
|
|
|
pkt1 = pkt->next;
|
|
|
|
|
av_free_packet(&pkt->pkt);
|
|
|
|
|
av_freep(&pkt);
|
|
|
|
|
}
|
|
|
|
|
q->last_pkt = NULL;
|
|
|
|
|
q->first_pkt = NULL;
|
|
|
|
|
q->size = 0;
|
|
|
|
|
SDL_UnlockMutex(q->mutex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
static void packet_queue_end(PacketQueue *q)
|
|
|
|
|
{
|
|
|
|
|
packet_queue_flush(q);
|
|
|
|
|
SDL_DestroyMutex(q->mutex);
|
|
|
|
|
SDL_DestroyCond(q->cond);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѹ<EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD> */
|
|
|
|
|
static int packet_queue_put(PacketQueue *q, AVPacket *pkt)
|
|
|
|
|
{
|
|
|
|
|
AVPacketList *pkt1;
|
|
|
|
|
|
|
|
|
|
pkt1 = av_malloc(sizeof(AVPacketList));
|
|
|
|
|
if (!pkt1)
|
|
|
|
|
return - 1;
|
|
|
|
|
pkt1->pkt = *pkt;
|
|
|
|
|
pkt1->next = NULL;
|
|
|
|
|
|
|
|
|
|
SDL_LockMutex(q->mutex);
|
|
|
|
|
|
|
|
|
|
if (!q->last_pkt)
|
|
|
|
|
q->first_pkt = pkt1;
|
|
|
|
|
else
|
|
|
|
|
q->last_pkt->next = pkt1;
|
|
|
|
|
q->last_pkt = pkt1;
|
|
|
|
|
q->size += pkt1->pkt.size;
|
|
|
|
|
|
|
|
|
|
SDL_CondSignal(q->cond);
|
|
|
|
|
|
|
|
|
|
SDL_UnlockMutex(q->mutex);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>˳<EFBFBD> */
|
|
|
|
|
static void packet_queue_abort(PacketQueue *q)
|
|
|
|
|
{
|
|
|
|
|
SDL_LockMutex(q->mutex);
|
|
|
|
|
|
|
|
|
|
q->abort_request = 1;
|
|
|
|
|
|
|
|
|
|
SDL_CondSignal(q->cond);
|
|
|
|
|
|
|
|
|
|
SDL_UnlockMutex(q->mutex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* return < 0 if aborted, 0 if no packet and > 0 if packet. */
|
|
|
|
|
/* <EFBFBD>Ӷ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ȡ<EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD> */
|
|
|
|
|
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; // <EFBFBD>쳣
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pkt1 = q->first_pkt;
|
|
|
|
|
if (pkt1)
|
|
|
|
|
{
|
|
|
|
|
q->first_pkt = pkt1->next;
|
|
|
|
|
if (!q->first_pkt)
|
|
|
|
|
q->last_pkt = NULL;
|
|
|
|
|
q->size -= pkt1->pkt.size;
|
|
|
|
|
*pkt = pkt1->pkt;
|
|
|
|
|
av_free(pkt1);
|
|
|
|
|
ret = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else if (!block)// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ǣ<EFBFBD>1(<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ģʽ)<EFBFBD><EFBFBD>0(<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ģʽ)
|
|
|
|
|
{
|
|
|
|
|
ret = 0; // <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ģʽ<EFBFBD><EFBFBD>û<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֱ<EFBFBD>ӷ<EFBFBD><EFBFBD><EFBFBD>0
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
SDL_CondWait(q->cond, q->mutex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
SDL_UnlockMutex(q->mutex);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD>VideoPicture<EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
static void alloc_picture(void *opaque)
|
|
|
|
|
{
|
|
|
|
|
VideoState *is = opaque;
|
|
|
|
|
VideoPicture *vp;
|
|
|
|
|
|
|
|
|
|
vp = &is->pictq[0];
|
|
|
|
|
|
|
|
|
|
if (vp->bmp)
|
|
|
|
|
SDL_FreeYUVOverlay(vp->bmp);
|
|
|
|
|
|
|
|
|
|
vp->bmp = SDL_CreateYUVOverlay(is->video_st->actx->width,
|
|
|
|
|
is->video_st->actx->height,
|
|
|
|
|
SDL_YV12_OVERLAY,
|
|
|
|
|
screen);
|
|
|
|
|
|
|
|
|
|
vp->width = is->video_st->actx->width;
|
|
|
|
|
vp->height = is->video_st->actx->height;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD>ʾͼ<EFBFBD><EFBFBD> */
|
|
|
|
|
static int video_display(VideoState *is, AVFrame *src_frame, double pts)
|
|
|
|
|
{
|
|
|
|
|
VideoPicture *vp;
|
|
|
|
|
int dst_pix_fmt;
|
|
|
|
|
AVPicture pict;
|
|
|
|
|
|
|
|
|
|
if (is->videoq.abort_request)
|
|
|
|
|
return - 1;
|
|
|
|
|
|
|
|
|
|
vp = &is->pictq[0];
|
|
|
|
|
|
|
|
|
|
/* if the frame is not skipped, then display it */
|
|
|
|
|
if (vp->bmp)
|
|
|
|
|
{
|
|
|
|
|
SDL_Rect rect;
|
|
|
|
|
|
|
|
|
|
if (pts)
|
|
|
|
|
Sleep((int)(is->frame_last_delay *1000));
|
|
|
|
|
#if 1
|
|
|
|
|
/* 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];
|
|
|
|
|
|
|
|
|
|
img_convert(&pict,
|
|
|
|
|
dst_pix_fmt,
|
|
|
|
|
(AVPicture*)src_frame,
|
|
|
|
|
is->video_st->actx->pix_fmt,
|
|
|
|
|
is->video_st->actx->width,
|
|
|
|
|
is->video_st->actx->height);
|
|
|
|
|
|
|
|
|
|
SDL_UnlockYUVOverlay(vp->bmp); /* update the bitmap content */
|
|
|
|
|
|
|
|
|
|
rect.x = 0;
|
|
|
|
|
rect.y = 0;
|
|
|
|
|
rect.w = is->video_st->actx->width;
|
|
|
|
|
rect.h = is->video_st->actx->height;
|
|
|
|
|
SDL_DisplayYUVOverlay(vp->bmp, &rect);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD> <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߽<EFBFBD><EFBFBD><EFBFBD>*/
|
|
|
|
|
static int video_thread(void *arg)
|
|
|
|
|
{
|
|
|
|
|
VideoState *is = arg;
|
|
|
|
|
AVPacket pkt1, *pkt = &pkt1;
|
|
|
|
|
int len1, got_picture;
|
|
|
|
|
double pts = 0;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD>֡
|
|
|
|
|
AVFrame *frame = av_malloc(sizeof(AVFrame));
|
|
|
|
|
memset(frame, 0, sizeof(AVFrame));
|
|
|
|
|
|
|
|
|
|
alloc_picture(is);
|
|
|
|
|
|
|
|
|
|
// ѭ<EFBFBD><EFBFBD>
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
// ȡ<EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD>
|
|
|
|
|
if (packet_queue_get(&is->videoq, pkt, 1) < 0)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ȼ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>룬Ȼ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
SDL_LockMutex(is->video_decoder_mutex);
|
|
|
|
|
len1 = avcodec_decode_video(is->video_st->actx, frame, &got_picture, pkt->data, pkt->size);
|
|
|
|
|
SDL_UnlockMutex(is->video_decoder_mutex);
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
if (pkt->dts != AV_NOPTS_VALUE)
|
|
|
|
|
pts = av_q2d(is->video_st->time_base) *pkt->dts;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>ʾ֡
|
|
|
|
|
if (got_picture)
|
|
|
|
|
{
|
|
|
|
|
if (video_display(is, frame, pts) < 0)
|
|
|
|
|
goto the_end;
|
|
|
|
|
}
|
|
|
|
|
av_free_packet(pkt);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
the_end:
|
|
|
|
|
av_free(frame);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* decode one audio frame and returns its uncompressed size */
|
|
|
|
|
static int audio_decode_frame(VideoState *is, uint8_t *audio_buf, double *pts_ptr)
|
|
|
|
|
{
|
|
|
|
|
AVPacket *pkt = &is->audio_pkt;
|
|
|
|
|
int len1, data_size;
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
/* NOTE: the audio packet can contain several frames */
|
|
|
|
|
while (is->audio_pkt_size > 0)
|
|
|
|
|
{
|
|
|
|
|
SDL_LockMutex(is->audio_decoder_mutex);
|
|
|
|
|
len1 = avcodec_decode_audio(is->audio_st->actx, (int16_t*)audio_buf,
|
|
|
|
|
&data_size, is->audio_pkt_data, is->audio_pkt_size);
|
|
|
|
|
|
|
|
|
|
SDL_UnlockMutex(is->audio_decoder_mutex);
|
|
|
|
|
if (len1 < 0)
|
|
|
|
|
{
|
|
|
|
|
/* if error, we skip the frame */
|
|
|
|
|
is->audio_pkt_size = 0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
is->audio_pkt_data += len1;
|
|
|
|
|
is->audio_pkt_size -= len1;
|
|
|
|
|
if (data_size <= 0)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
return data_size;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* free the current packet */
|
|
|
|
|
if (pkt->data)
|
|
|
|
|
av_free_packet(pkt);
|
|
|
|
|
|
|
|
|
|
/* read next packet */
|
|
|
|
|
if (packet_queue_get(&is->audioq, pkt, 1) < 0)
|
|
|
|
|
return - 1;
|
|
|
|
|
|
|
|
|
|
is->audio_pkt_data = pkt->data;
|
|
|
|
|
is->audio_pkt_size = pkt->size;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* prepare a new audio buffer */
|
|
|
|
|
void sdl_audio_callback(void *opaque, Uint8 *stream, int len)
|
|
|
|
|
{
|
|
|
|
|
VideoState *is = opaque;
|
|
|
|
|
int audio_size, len1;
|
|
|
|
|
double pts = 0;
|
|
|
|
|
|
|
|
|
|
while (len > 0)
|
|
|
|
|
{
|
|
|
|
|
if (is->audio_buf_index >= is->audio_buf_size)
|
|
|
|
|
{
|
|
|
|
|
audio_size = audio_decode_frame(is, is->audio_buf, &pts);
|
|
|
|
|
if (audio_size < 0)
|
|
|
|
|
{
|
|
|
|
|
/* if error, just output silence */
|
|
|
|
|
is->audio_buf_size = 1024;
|
|
|
|
|
memset(is->audio_buf, 0, is->audio_buf_size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// audio_size = synchronize_audio(is, (int16_t*)is->audio_buf, audio_size, 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 */
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD> <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>ȥ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
enc = ic->streams[stream_index]->actx;
|
|
|
|
|
|
|
|
|
|
/* prepare audio output */
|
|
|
|
|
/* <EFBFBD><EFBFBD>Ƶ */
|
|
|
|
|
if (enc->codec_type == CODEC_TYPE_AUDIO)
|
|
|
|
|
{
|
|
|
|
|
wanted_spec.freq = enc->sample_rate;
|
|
|
|
|
wanted_spec.format = AUDIO_S16SYS;
|
|
|
|
|
/* hack for AC3. XXX: suppress that */
|
|
|
|
|
if (enc->channels > 2)
|
|
|
|
|
enc->channels = 2;
|
|
|
|
|
wanted_spec.channels = enc->channels;
|
|
|
|
|
wanted_spec.silence = 0;
|
|
|
|
|
wanted_spec.samples = 1024; //SDL_AUDIO_BUFFER_SIZE;
|
|
|
|
|
wanted_spec.callback = sdl_audio_callback;
|
|
|
|
|
wanted_spec.userdata = is;
|
|
|
|
|
if (SDL_OpenAudio(&wanted_spec, &spec) < 0)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "SDL_OpenAudio: %s\n", SDL_GetError());
|
|
|
|
|
return - 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
codec = avcodec_find_decoder(enc->codec_id);
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
if (!codec || avcodec_open(enc, codec) < 0)
|
|
|
|
|
return - 1;
|
|
|
|
|
|
|
|
|
|
switch (enc->codec_type)
|
|
|
|
|
{
|
|
|
|
|
case CODEC_TYPE_AUDIO: // <EFBFBD><EFBFBD>Ƶ
|
|
|
|
|
is->audio_stream = stream_index;
|
|
|
|
|
is->audio_st = ic->streams[stream_index];
|
|
|
|
|
is->audio_buf_size = 0;
|
|
|
|
|
is->audio_buf_index = 0;
|
|
|
|
|
|
|
|
|
|
memset(&is->audio_pkt, 0, sizeof(is->audio_pkt));
|
|
|
|
|
packet_queue_init(&is->audioq);
|
|
|
|
|
SDL_PauseAudio(0);
|
|
|
|
|
break;
|
|
|
|
|
case CODEC_TYPE_VIDEO:
|
|
|
|
|
is->video_stream = stream_index;
|
|
|
|
|
is->video_st = ic->streams[stream_index];
|
|
|
|
|
|
|
|
|
|
is->frame_last_delay = is->video_st->frame_last_delay;
|
|
|
|
|
|
|
|
|
|
packet_queue_init(&is->videoq); // <EFBFBD><EFBFBD><EFBFBD>г<EFBFBD>ʼ<EFBFBD><EFBFBD>
|
|
|
|
|
is->video_tid = SDL_CreateThread(video_thread, is); // <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƶ<EFBFBD>߳<EFBFBD>ȥ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void stream_component_close(VideoState *is, int stream_index)
|
|
|
|
|
{
|
|
|
|
|
AVFormatContext *ic = is->ic;
|
|
|
|
|
AVCodecContext *enc;
|
|
|
|
|
|
|
|
|
|
if (stream_index < 0 || stream_index >= ic->nb_streams)
|
|
|
|
|
return ;
|
|
|
|
|
enc = ic->streams[stream_index]->actx;
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
SDL_WaitThread(is->video_tid, NULL);
|
|
|
|
|
packet_queue_end(&is->videoq);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avcodec_close(enc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD> */
|
|
|
|
|
static int decode_thread(void *arg)
|
|
|
|
|
{
|
|
|
|
|
VideoState *is = arg;
|
|
|
|
|
AVFormatContext *ic;
|
|
|
|
|
int err, i, ret, video_index, audio_index;
|
|
|
|
|
AVPacket pkt1, *pkt = &pkt1;
|
|
|
|
|
AVFormatParameters params, *ap = ¶ms;
|
|
|
|
|
|
|
|
|
|
int flags = SDL_HWSURFACE | SDL_ASYNCBLIT | SDL_HWACCEL | SDL_RESIZABLE;
|
|
|
|
|
|
|
|
|
|
video_index = - 1;
|
|
|
|
|
audio_index = - 1;
|
|
|
|
|
|
|
|
|
|
is->video_stream = - 1;
|
|
|
|
|
is->audio_stream = - 1;
|
|
|
|
|
|
|
|
|
|
memset(ap, 0, sizeof(*ap));
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ļ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD>䵽IO<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ķ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
err = av_open_input_file(&ic, is->filename, NULL, 0, ap);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
{
|
|
|
|
|
ret = - 1;
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
is->ic = ic;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
for (i = 0; i < ic->nb_streams; i++)
|
|
|
|
|
{
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
AVCodecContext *enc = ic->streams[i]->actx;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
switch (enc->codec_type)
|
|
|
|
|
{
|
|
|
|
|
case CODEC_TYPE_AUDIO: // <EFBFBD><EFBFBD>Ƶ
|
|
|
|
|
if (audio_index < 0)
|
|
|
|
|
audio_index = i;
|
|
|
|
|
break;
|
|
|
|
|
case CODEC_TYPE_VIDEO: // <EFBFBD><EFBFBD>Ƶ
|
|
|
|
|
if (video_index < 0)
|
|
|
|
|
video_index = i;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƶģʽ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>õ<EFBFBD><EFBFBD><EFBFBD>Ļ
|
|
|
|
|
screen = SDL_SetVideoMode(enc->width, enc->height, 0, flags);
|
|
|
|
|
|
|
|
|
|
SDL_WM_SetCaption("FFplay", "FFplay"); // <EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϊ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>С
|
|
|
|
|
|
|
|
|
|
// schedule_refresh(is, 40);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ijɷ֣<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>÷<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>
|
|
|
|
|
if (audio_index >= 0)
|
|
|
|
|
stream_component_open(is, audio_index);
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ĸ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ɷ֣<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>÷<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>
|
|
|
|
|
if (video_index >= 0)
|
|
|
|
|
stream_component_open(is, video_index);
|
|
|
|
|
|
|
|
|
|
if (is->video_stream < 0 && is->audio_stream < 0)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "%s: could not open codecs\n", is->filename);
|
|
|
|
|
ret = - 1;
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѭ<EFBFBD><EFBFBD>
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
if (is->abort_request)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ѿ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ˣ<EFBFBD><EFBFBD><EFBFBD>ô<EFBFBD>ӳ<EFBFBD>ȥ<EFBFBD><EFBFBD>
|
|
|
|
|
if (is->audioq.size > MAX_AUDIOQ_SIZE || is->videoq.size > MAX_VIDEOQ_SIZE || url_feof(&ic->pb))
|
|
|
|
|
{
|
|
|
|
|
SDL_Delay(10); // if the queue are full, no need to read more,wait 10 ms
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>ȡһ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʾһ֡<EFBFBD><EFBFBD>
|
|
|
|
|
ret = av_read_packet(ic, pkt); //av_read_frame(ic, pkt);
|
|
|
|
|
if (ret < 0)
|
|
|
|
|
{
|
|
|
|
|
if (url_ferror(&ic->pb) == 0)
|
|
|
|
|
{
|
|
|
|
|
SDL_Delay(100); // wait for user event
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݰ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ӵ<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
av_free_packet(pkt);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (!is->abort_request) // wait until the end
|
|
|
|
|
{
|
|
|
|
|
SDL_Delay(100);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
if (is->audio_stream >= 0)
|
|
|
|
|
stream_component_close(is, is->audio_stream);
|
|
|
|
|
|
|
|
|
|
if (is->video_stream >= 0)
|
|
|
|
|
stream_component_close(is, is->video_stream);
|
|
|
|
|
|
|
|
|
|
if (is->ic)
|
|
|
|
|
{
|
|
|
|
|
av_close_input_file(is->ic);
|
|
|
|
|
is->ic = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ret != 0)
|
|
|
|
|
{
|
|
|
|
|
SDL_Event event;
|
|
|
|
|
|
|
|
|
|
event.type = FF_QUIT_EVENT;
|
|
|
|
|
event.user.data1 = is;
|
|
|
|
|
SDL_PushEvent(&event);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ļ<EFBFBD> <EFBFBD><EFBFBD>*/
|
|
|
|
|
static VideoState *stream_open(const char *filename, AVInputFormat *iformat)
|
|
|
|
|
{
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ״̬
|
|
|
|
|
VideoState *is;
|
|
|
|
|
|
|
|
|
|
is = av_mallocz(sizeof(VideoState));
|
|
|
|
|
if (!is)
|
|
|
|
|
return NULL;
|
|
|
|
|
pstrcpy(is->filename, sizeof(is->filename), filename);
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>
|
|
|
|
|
is->audio_decoder_mutex = SDL_CreateMutex();
|
|
|
|
|
// <EFBFBD><EFBFBD>Ƶ<EFBFBD><EFBFBD>
|
|
|
|
|
is->video_decoder_mutex = SDL_CreateMutex();
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>߳<EFBFBD>
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
is->abort_request = 1;
|
|
|
|
|
SDL_WaitThread(is->parse_tid, NULL);
|
|
|
|
|
|
|
|
|
|
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->audio_decoder_mutex);
|
|
|
|
|
SDL_DestroyMutex(is->video_decoder_mutex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void do_exit(void)
|
|
|
|
|
{
|
|
|
|
|
if (cur_stream)
|
|
|
|
|
{
|
|
|
|
|
stream_close(cur_stream);
|
|
|
|
|
cur_stream = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SDL_Quit();
|
|
|
|
|
exit(0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void event_loop(void) // handle an event sent by the GUI
|
|
|
|
|
{
|
|
|
|
|
SDL_Event event;
|
|
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
SDL_WaitEvent(&event);
|
|
|
|
|
switch (event.type)
|
|
|
|
|
{
|
|
|
|
|
case SDL_KEYDOWN:
|
|
|
|
|
switch (event.key.keysym.sym)
|
|
|
|
|
{
|
|
|
|
|
case SDLK_ESCAPE:
|
|
|
|
|
case SDLK_q:
|
|
|
|
|
do_exit();
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case SDL_QUIT:
|
|
|
|
|
case FF_QUIT_EVENT:
|
|
|
|
|
do_exit();
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
|
{
|
|
|
|
|
int flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER;
|
|
|
|
|
|
|
|
|
|
// ע<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>֧<EFBFBD>ֵĸ<EFBFBD>ʽ
|
|
|
|
|
av_register_all();
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ļ<EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
input_filename = "clocktxt_320.avi";
|
|
|
|
|
// input_filename = "d:/yuv/clocktxt.avi";
|
|
|
|
|
|
|
|
|
|
// sdl<EFBFBD><EFBFBD>ʼ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>sdl<EFBFBD><EFBFBD>һ<EFBFBD><EFBFBD><EFBFBD><EFBFBD>ƽ̨<EFBFBD><EFBFBD>ý<EFBFBD>忪<EFBFBD><EFBFBD><EFBFBD>⣩
|
|
|
|
|
if (SDL_Init(flags))
|
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
|
|
// ע<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>¼<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>¼<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>¼<EFBFBD><EFBFBD><EFBFBD>windows<EFBFBD><EFBFBD>ϵͳ<EFBFBD><EFBFBD>Ϣ<EFBFBD><EFBFBD><EFBFBD>û<EFBFBD><EFBFBD>¼<EFBFBD>
|
|
|
|
|
SDL_EventState(SDL_ACTIVEEVENT, SDL_IGNORE);
|
|
|
|
|
SDL_EventState(SDL_MOUSEMOTION, SDL_IGNORE);
|
|
|
|
|
SDL_EventState(SDL_SYSWMEVENT, SDL_IGNORE);
|
|
|
|
|
SDL_EventState(SDL_USEREVENT, SDL_IGNORE);
|
|
|
|
|
|
|
|
|
|
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
cur_stream = stream_open(input_filename, file_iformat);
|
|
|
|
|
|
|
|
|
|
event_loop();
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|