yading@10
|
1 /*
|
yading@10
|
2 * MJPEG parser
|
yading@10
|
3 * Copyright (c) 2000, 2001 Fabrice Bellard
|
yading@10
|
4 * Copyright (c) 2003 Alex Beregszaszi
|
yading@10
|
5 * Copyright (c) 2003-2004 Michael Niedermayer
|
yading@10
|
6 *
|
yading@10
|
7 * This file is part of FFmpeg.
|
yading@10
|
8 *
|
yading@10
|
9 * FFmpeg is free software; you can redistribute it and/or
|
yading@10
|
10 * modify it under the terms of the GNU Lesser General Public
|
yading@10
|
11 * License as published by the Free Software Foundation; either
|
yading@10
|
12 * version 2.1 of the License, or (at your option) any later version.
|
yading@10
|
13 *
|
yading@10
|
14 * FFmpeg is distributed in the hope that it will be useful,
|
yading@10
|
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
yading@10
|
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
yading@10
|
17 * Lesser General Public License for more details.
|
yading@10
|
18 *
|
yading@10
|
19 * You should have received a copy of the GNU Lesser General Public
|
yading@10
|
20 * License along with FFmpeg; if not, write to the Free Software
|
yading@10
|
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
yading@10
|
22 */
|
yading@10
|
23
|
yading@10
|
24 /**
|
yading@10
|
25 * @file
|
yading@10
|
26 * MJPEG parser.
|
yading@10
|
27 */
|
yading@10
|
28
|
yading@10
|
29 #include "parser.h"
|
yading@10
|
30
|
yading@10
|
31 typedef struct MJPEGParserContext{
|
yading@10
|
32 ParseContext pc;
|
yading@10
|
33 int size;
|
yading@10
|
34 }MJPEGParserContext;
|
yading@10
|
35
|
yading@10
|
36 /**
|
yading@10
|
37 * Find the end of the current frame in the bitstream.
|
yading@10
|
38 * @return the position of the first byte of the next frame, or -1
|
yading@10
|
39 */
|
yading@10
|
40 static int find_frame_end(MJPEGParserContext *m, const uint8_t *buf, int buf_size){
|
yading@10
|
41 ParseContext *pc= &m->pc;
|
yading@10
|
42 int vop_found, i;
|
yading@10
|
43 uint32_t state;
|
yading@10
|
44
|
yading@10
|
45 vop_found= pc->frame_start_found;
|
yading@10
|
46 state= pc->state;
|
yading@10
|
47
|
yading@10
|
48 i=0;
|
yading@10
|
49 if(!vop_found){
|
yading@10
|
50 for(i=0; i<buf_size;){
|
yading@10
|
51 state= (state<<8) | buf[i];
|
yading@10
|
52 if(state>=0xFFC00000 && state<=0xFFFEFFFF){
|
yading@10
|
53 if(state>=0xFFD80000 && state<=0xFFD8FFFF){
|
yading@10
|
54 i++;
|
yading@10
|
55 vop_found=1;
|
yading@10
|
56 break;
|
yading@10
|
57 }else if(state<0xFFD00000 || state>0xFFD9FFFF){
|
yading@10
|
58 m->size= (state&0xFFFF)-1;
|
yading@10
|
59 }
|
yading@10
|
60 }
|
yading@10
|
61 if(m->size>0){
|
yading@10
|
62 int size= FFMIN(buf_size-i, m->size);
|
yading@10
|
63 i+=size;
|
yading@10
|
64 m->size-=size;
|
yading@10
|
65 state=0;
|
yading@10
|
66 continue;
|
yading@10
|
67 }else
|
yading@10
|
68 i++;
|
yading@10
|
69 }
|
yading@10
|
70 }
|
yading@10
|
71
|
yading@10
|
72 if(vop_found){
|
yading@10
|
73 /* EOF considered as end of frame */
|
yading@10
|
74 if (buf_size == 0)
|
yading@10
|
75 return 0;
|
yading@10
|
76 for(; i<buf_size;){
|
yading@10
|
77 state= (state<<8) | buf[i];
|
yading@10
|
78 if(state>=0xFFC00000 && state<=0xFFFEFFFF){
|
yading@10
|
79 if(state>=0xFFD80000 && state<=0xFFD8FFFF){
|
yading@10
|
80 pc->frame_start_found=0;
|
yading@10
|
81 pc->state=0;
|
yading@10
|
82 return i-3;
|
yading@10
|
83 } else if(state<0xFFD00000 || state>0xFFD9FFFF){
|
yading@10
|
84 m->size= (state&0xFFFF)-1;
|
yading@10
|
85 }
|
yading@10
|
86 }
|
yading@10
|
87 if(m->size>0){
|
yading@10
|
88 int size= FFMIN(buf_size-i, m->size);
|
yading@10
|
89 i+=size;
|
yading@10
|
90 m->size-=size;
|
yading@10
|
91 state=0;
|
yading@10
|
92 continue;
|
yading@10
|
93 }else
|
yading@10
|
94 i++;
|
yading@10
|
95 }
|
yading@10
|
96 }
|
yading@10
|
97 pc->frame_start_found= vop_found;
|
yading@10
|
98 pc->state= state;
|
yading@10
|
99 return END_NOT_FOUND;
|
yading@10
|
100 }
|
yading@10
|
101
|
yading@10
|
102 static int jpeg_parse(AVCodecParserContext *s,
|
yading@10
|
103 AVCodecContext *avctx,
|
yading@10
|
104 const uint8_t **poutbuf, int *poutbuf_size,
|
yading@10
|
105 const uint8_t *buf, int buf_size)
|
yading@10
|
106 {
|
yading@10
|
107 MJPEGParserContext *m = s->priv_data;
|
yading@10
|
108 ParseContext *pc = &m->pc;
|
yading@10
|
109 int next;
|
yading@10
|
110
|
yading@10
|
111 if(s->flags & PARSER_FLAG_COMPLETE_FRAMES){
|
yading@10
|
112 next= buf_size;
|
yading@10
|
113 }else{
|
yading@10
|
114 next= find_frame_end(m, buf, buf_size);
|
yading@10
|
115
|
yading@10
|
116 if (ff_combine_frame(pc, next, &buf, &buf_size) < 0) {
|
yading@10
|
117 *poutbuf = NULL;
|
yading@10
|
118 *poutbuf_size = 0;
|
yading@10
|
119 return buf_size;
|
yading@10
|
120 }
|
yading@10
|
121 }
|
yading@10
|
122
|
yading@10
|
123 *poutbuf = buf;
|
yading@10
|
124 *poutbuf_size = buf_size;
|
yading@10
|
125 return next;
|
yading@10
|
126 }
|
yading@10
|
127
|
yading@10
|
128
|
yading@10
|
129 AVCodecParser ff_mjpeg_parser = {
|
yading@10
|
130 .codec_ids = { AV_CODEC_ID_MJPEG },
|
yading@10
|
131 .priv_data_size = sizeof(MJPEGParserContext),
|
yading@10
|
132 .parser_parse = jpeg_parse,
|
yading@10
|
133 .parser_close = ff_parse_close,
|
yading@10
|
134 };
|